summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--configure.ac6
-rw-r--r--ext/jp2k/gstjasperenc.c6
-rw-r--r--gst/deinterlace/gstdeinterlace.c3
-rw-r--r--gst/mxf/Makefile.am10
-rw-r--r--gst/mxf/mxf.c8
-rw-r--r--gst/mxf/mxfaes-bwf.c533
-rw-r--r--gst/mxf/mxfalaw.c160
-rw-r--r--gst/mxf/mxfdemux.c2
-rw-r--r--gst/mxf/mxfdv-dif.c98
-rw-r--r--gst/mxf/mxfjpeg2000.c121
-rw-r--r--gst/mxf/mxfmetadata.c2296
-rw-r--r--gst/mxf/mxfmetadata.h6
-rw-r--r--gst/mxf/mxfmpeg.c555
-rw-r--r--gst/mxf/mxfmux.c1429
-rw-r--r--gst/mxf/mxfmux.h103
-rw-r--r--gst/mxf/mxfparse.c289
-rw-r--r--gst/mxf/mxfparse.h12
-rw-r--r--gst/mxf/mxftypes.h8
-rw-r--r--gst/mxf/mxful.c268
-rw-r--r--gst/mxf/mxful.h59
-rw-r--r--gst/mxf/mxfup.c451
-rw-r--r--gst/mxf/mxfvc3.c87
-rw-r--r--gst/mxf/mxfwrite.c567
-rw-r--r--gst/mxf/mxfwrite.h85
24 files changed, 6867 insertions, 295 deletions
diff --git a/configure.ac b/configure.ac
index 0df70751..245e3e87 100644
--- a/configure.ac
+++ b/configure.ac
@@ -45,7 +45,7 @@ AC_LIBTOOL_WIN32_DLL
AM_PROG_LIBTOOL
dnl *** required versions of GStreamer stuff ***
-GST_REQ=0.10.22
+GST_REQ=0.10.22.1
GSTPB_REQ=0.10.22
dnl *** autotools stuff ****
@@ -160,6 +160,10 @@ fi
AM_CONDITIONAL(HAVE_GCC_ASM, test "x$HAVE_GCC_ASM" = "xyes")
dnl *** checks for library functions ***
+AC_CHECK_FUNCS([gmtime_r])
+
+dnl *** checks for headers ***
+AC_CHECK_HEADERS([sys/utsname.h])
dnl *** checks for dependency libraries ***
diff --git a/ext/jp2k/gstjasperenc.c b/ext/jp2k/gstjasperenc.c
index a81a2a93..fbd4fd57 100644
--- a/ext/jp2k/gstjasperenc.c
+++ b/ext/jp2k/gstjasperenc.c
@@ -62,9 +62,11 @@ static GstStaticPadTemplate gst_jasper_enc_src_template =
GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_SRC,
GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("image/x-j2c, "
+ GST_STATIC_CAPS ("image/x-j2c, width = " GST_VIDEO_SIZE_RANGE ", height = "
+ GST_VIDEO_SIZE_RANGE ", fourcc = (GstFourcc) { sRGB, sYUV },"
"framerate = " GST_VIDEO_FPS_RANGE ", " "fields = (int) 1; "
- "image/x-jpc, "
+ "image/x-jpc, width = " GST_VIDEO_SIZE_RANGE ", height = "
+ GST_VIDEO_SIZE_RANGE ", fourcc = (GstFourcc) { sRGB, sYUV },"
"framerate = " GST_VIDEO_FPS_RANGE ", " "fields = (int) 1; "
"image/jp2")
);
diff --git a/gst/deinterlace/gstdeinterlace.c b/gst/deinterlace/gstdeinterlace.c
index 70fe1420..b2a25afe 100644
--- a/gst/deinterlace/gstdeinterlace.c
+++ b/gst/deinterlace/gstdeinterlace.c
@@ -353,6 +353,9 @@ gst_deinterlace_transform_ip (GstBaseTransform * trans, GstBuffer * buf)
height = filter->uv_height;
fill_value = 128;
break;
+ default:
+ g_assert_not_reached ();
+ break;
}
for (x = 0; x < width; x++) {
diff --git a/gst/mxf/Makefile.am b/gst/mxf/Makefile.am
index ed824355..96534c4b 100644
--- a/gst/mxf/Makefile.am
+++ b/gst/mxf/Makefile.am
@@ -14,7 +14,10 @@ libgstmxf_la_SOURCES = \
mxfup.c \
mxfvc3.c \
mxfmetadata.c \
- mxfdms1.c
+ mxfdms1.c \
+ mxfwrite.c \
+ mxfmux.c \
+ mxful.c
libgstmxf_la_CFLAGS = $(GST_CFLAGS) $(GST_BASE_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS)
libgstmxf_la_LIBADD = $(GST_LIBS) $(GST_BASE_LIBS) $(GST_PLUGINS_BASE_LIBS) \
@@ -36,5 +39,8 @@ noinst_HEADERS = \
mxfvc3.h \
mxftypes.h \
mxfmetadata.h \
- mxfdms1.h
+ mxfdms1.h \
+ mxfwrite.h \
+ mxfmux.h \
+ mxful.h
diff --git a/gst/mxf/mxf.c b/gst/mxf/mxf.c
index ec477721..a906f1d8 100644
--- a/gst/mxf/mxf.c
+++ b/gst/mxf/mxf.c
@@ -24,6 +24,7 @@
#include "mxfquark.h"
#include "mxfdemux.h"
+#include "mxfmux.h"
#include "mxfaes-bwf.h"
#include "mxfmpeg.h"
#include "mxfdv-dif.h"
@@ -53,6 +54,8 @@ mxf_init (void)
static gboolean
plugin_init (GstPlugin * plugin)
{
+ GST_DEBUG_CATEGORY_INIT (mxf_debug, "mxf", 0, "MXF");
+
mxf_init ();
mxf_quark_initialize ();
mxf_metadata_init_types ();
@@ -67,11 +70,10 @@ plugin_init (GstPlugin * plugin)
mxf_dms1_initialize ();
if (!gst_element_register (plugin, "mxfdemux", GST_RANK_PRIMARY,
- GST_TYPE_MXF_DEMUX))
+ GST_TYPE_MXF_DEMUX) ||
+ !gst_element_register (plugin, "mxfmux", GST_RANK_NONE, GST_TYPE_MXF_MUX))
return FALSE;
- GST_DEBUG_CATEGORY_INIT (mxf_debug, "mxf", 0, "MXF");
-
return TRUE;
}
diff --git a/gst/mxf/mxfaes-bwf.c b/gst/mxf/mxfaes-bwf.c
index d40ff6a6..df9c0047 100644
--- a/gst/mxf/mxfaes-bwf.c
+++ b/gst/mxf/mxfaes-bwf.c
@@ -35,6 +35,7 @@
#include <gst/gst.h>
#include <string.h>
+#include "mxfwrite.h"
#include "mxfaes-bwf.h"
#include "mxfquark.h"
@@ -305,6 +306,210 @@ mxf_metadata_wave_audio_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_wave_audio_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataWaveAudioEssenceDescriptor *self =
+ MXF_METADATA_WAVE_AUDIO_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_wave_audio_essence_descriptor_parent_class)->write_tags (m,
+ primer);
+ MXFLocalTag *t;
+ static const guint8 block_align_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x03, 0x02, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 sequence_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x03, 0x02, 0x02, 0x00, 0x00, 0x00
+ };
+ static const guint8 avg_bps_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x03, 0x03, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 channel_assignment_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x07,
+ 0x04, 0x02, 0x01, 0x01, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_envelope_version_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x06, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_envelope_format_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x07, 0x00, 0x00, 0x00
+ };
+ static const guint8 points_per_peak_value_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x08, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_envelope_block_size_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x09, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_channels_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x0A, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_frames_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x0B, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_of_peaks_position_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x0C, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_envelope_timestamp_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x0D, 0x00, 0x00, 0x00
+ };
+ static const guint8 peak_envelope_data_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x08,
+ 0x04, 0x02, 0x03, 0x01, 0x0E, 0x00, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &block_align_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->block_align);
+ mxf_primer_pack_add_mapping (primer, 0x3d0a, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->sequence_offset) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sequence_offset_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->sequence_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3d0b, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &avg_bps_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->avg_bps);
+ mxf_primer_pack_add_mapping (primer, 0x3d09, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (!mxf_ul_is_zero (&self->channel_assignment)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &channel_assignment_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->channel_assignment, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3d32, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_envelope_version) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_envelope_version_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->peak_envelope_version);
+ mxf_primer_pack_add_mapping (primer, 0x3d29, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_envelope_format) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_envelope_format_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->peak_envelope_format);
+ mxf_primer_pack_add_mapping (primer, 0x3d2a, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->points_per_peak_value) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &points_per_peak_value_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->points_per_peak_value);
+ mxf_primer_pack_add_mapping (primer, 0x3d2b, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_envelope_block_size) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_envelope_block_size_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->peak_envelope_block_size);
+ mxf_primer_pack_add_mapping (primer, 0x3d2c, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_channels) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_channels_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->peak_channels);
+ mxf_primer_pack_add_mapping (primer, 0x3d2d, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_frames) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_frames_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->peak_frames);
+ mxf_primer_pack_add_mapping (primer, 0x3d2e, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_of_peaks_position) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_of_peaks_position_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->peak_of_peaks_position);
+ mxf_primer_pack_add_mapping (primer, 0x3d2f, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_timestamp_is_unknown (&self->peak_envelope_timestamp)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_envelope_timestamp_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_timestamp_write (&self->peak_envelope_timestamp, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x3d30, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->peak_envelope_data) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &peak_envelope_data_ul, 16);
+ t->size = self->peak_envelope_data_length;
+ t->data = g_memdup (self->peak_envelope_data, t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3d31, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_wave_audio_essence_descriptor_init
(MXFMetadataWaveAudioEssenceDescriptor * self)
@@ -324,6 +529,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (WAVE_AUDIO_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_wave_audio_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_wave_audio_essence_descriptor_write_tags;
metadata_class->type = 0x0148;
}
@@ -705,6 +912,139 @@ mxf_metadata_aes3_audio_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_aes3_audio_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataAES3AudioEssenceDescriptor *self =
+ MXF_METADATA_AES3_AUDIO_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_aes3_audio_essence_descriptor_parent_class)->write_tags (m,
+ primer);
+ MXFLocalTag *t;
+ static const guint8 emphasis_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x06, 0x00, 0x00, 0x00
+ };
+ static const guint8 block_start_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x03, 0x02, 0x03, 0x00, 0x00, 0x00
+ };
+ static const guint8 auxiliary_bits_mode_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 channel_status_mode_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x02, 0x00, 0x00, 0x00
+ };
+ static const guint8 fixed_channel_status_data_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x03, 0x00, 0x00, 0x00
+ };
+ static const guint8 user_data_mode_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 fixed_user_data_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x05, 0x01, 0x05, 0x00, 0x00, 0x00
+ };
+
+ if (self->emphasis) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &emphasis_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->emphasis);
+ mxf_primer_pack_add_mapping (primer, 0x3d0d, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->block_start_offset) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &block_start_offset_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->block_start_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3d0f, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->auxiliary_bits_mode) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &auxiliary_bits_mode_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->auxiliary_bits_mode);
+ mxf_primer_pack_add_mapping (primer, 0x3d08, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->channel_status_mode) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &channel_status_mode_ul, 16);
+ t->size = 8 + self->n_channel_status_mode;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_channel_status_mode);
+ GST_WRITE_UINT32_BE (t->data + 4, 1);
+ memcpy (t->data + 8, self->channel_status_mode, t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3d10, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->fixed_channel_status_data) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &fixed_channel_status_data_ul, 16);
+ t->size = 8 + 24 * self->n_fixed_channel_status_data;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_fixed_channel_status_data);
+ GST_WRITE_UINT32_BE (t->data + 4, 24);
+ for (i = 0; i < self->n_fixed_channel_status_data; i++)
+ memcpy (t->data + 8 + 24 * i, self->fixed_channel_status_data[i], 24);
+ mxf_primer_pack_add_mapping (primer, 0x3d11, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->user_data_mode) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &user_data_mode_ul, 16);
+ t->size = 8 + self->n_user_data_mode;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_user_data_mode);
+ GST_WRITE_UINT32_BE (t->data + 4, 1);
+ memcpy (t->data + 8, self->user_data_mode, t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3d12, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->fixed_user_data) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &fixed_user_data_ul, 16);
+ t->size = 8 + 24 * self->n_fixed_user_data;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_fixed_user_data);
+ GST_WRITE_UINT32_BE (t->data + 4, 24);
+ for (i = 0; i < self->n_fixed_user_data; i++)
+ memcpy (t->data + 8 + 24 * i, self->fixed_user_data[i], 24);
+ mxf_primer_pack_add_mapping (primer, 0x3d11, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
static void
mxf_metadata_aes3_audio_essence_descriptor_init
@@ -728,6 +1068,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (AES3_AUDIO_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_aes3_audio_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_aes3_audio_essence_descriptor_write_tags;
metadata_class->type = 0x0147;
}
@@ -1051,6 +1393,190 @@ static const MXFEssenceElementHandler mxf_aes_bwf_essence_handler = {
mxf_aes_bwf_create_caps
};
+typedef struct
+{
+ guint64 error;
+ gint width, rate, channels;
+ MXFFraction edit_rate;
+} BWFMappingData;
+
+static GstFlowReturn
+mxf_bwf_write_func (GstBuffer * buffer, GstCaps * caps, gpointer mapping_data,
+ GstAdapter * adapter, GstBuffer ** outbuf, gboolean flush)
+{
+ BWFMappingData *md = mapping_data;
+ guint bytes;
+ guint64 speu =
+ gst_util_uint64_scale (md->rate, md->edit_rate.d, md->edit_rate.n);
+
+ md->error += (md->edit_rate.d * md->rate) % (md->edit_rate.n);
+ if (md->error >= md->edit_rate.n) {
+ md->error = 0;
+ speu += 1;
+ }
+
+ bytes = (speu * md->channels * md->width) / 8;
+
+ if (buffer)
+ gst_adapter_push (adapter, buffer);
+
+ if (gst_adapter_available (adapter) == 0)
+ return GST_FLOW_OK;
+
+ if (flush)
+ bytes = MIN (gst_adapter_available (adapter), bytes);
+
+ if (gst_adapter_available (adapter) >= bytes) {
+ *outbuf = gst_adapter_take_buffer (adapter, bytes);
+ }
+
+ if (gst_adapter_available (adapter) >= bytes)
+ return GST_FLOW_CUSTOM_SUCCESS;
+ else
+ return GST_FLOW_OK;
+}
+
+static const guint8 bwf_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x01,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x06, 0x01, 0x00
+};
+
+static MXFMetadataFileDescriptor *
+mxf_bwf_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataWaveAudioEssenceDescriptor *ret;
+ GstStructure *s;
+ BWFMappingData *md;
+ gint width, rate, channels, endianness;
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "audio/x-raw-int") != 0 ||
+ !gst_structure_get_int (s, "width", &width) ||
+ !gst_structure_get_int (s, "rate", &rate) ||
+ !gst_structure_get_int (s, "channels", &channels) ||
+ !gst_structure_get_int (s, "endianness", &endianness)) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ return NULL;
+ }
+
+ ret = (MXFMetadataWaveAudioEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_WAVE_AUDIO_ESSENCE_DESCRIPTOR);
+
+ memcpy (&ret->parent.parent.essence_container, &bwf_essence_container_ul, 16);
+ if (endianness == G_LITTLE_ENDIAN)
+ memcpy (&ret->parent.sound_essence_compression,
+ &mxf_sound_essence_compression_uncompressed, 16);
+ else
+ memcpy (&ret->parent.sound_essence_compression,
+ &mxf_sound_essence_compression_aiff, 16);
+
+ ret->block_align = (width / 8) * channels;
+ ret->parent.quantization_bits = width;
+ ret->avg_bps = ret->block_align * rate;
+
+ if (!mxf_metadata_generic_sound_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_bwf_write_func;
+
+ md = g_new0 (BWFMappingData, 1);
+ md->width = width;
+ md->rate = rate;
+ md->channels = channels;
+ *mapping_data = md;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_bwf_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_bwf_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ guint i;
+ gdouble min = G_MAXDOUBLE;
+ BWFMappingData *md = mapping_data;
+
+ for (i = 0; i < package->parent.n_tracks; i++) {
+ MXFMetadataTimelineTrack *tmp;
+
+ if (!MXF_IS_METADATA_TIMELINE_TRACK (package->parent.tracks[i]) ||
+ package->parent.tracks[i] == (MXFMetadataTrack *) track)
+ continue;
+
+ tmp = MXF_METADATA_TIMELINE_TRACK (package->parent.tracks[i]);
+ if (((gdouble) tmp->edit_rate.n) / ((gdouble) tmp->edit_rate.d) < min) {
+ min = ((gdouble) tmp->edit_rate.n) / ((gdouble) tmp->edit_rate.d);
+ memcpy (edit_rate, &tmp->edit_rate, sizeof (MXFFraction));
+ }
+ }
+
+ if (min == G_MAXDOUBLE) {
+ /* 100ms edit units */
+ edit_rate->n = 10;
+ edit_rate->d = 1;
+ }
+
+ memcpy (&md->edit_rate, edit_rate, sizeof (MXFFraction));
+}
+
+static guint32
+mxf_bwf_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x16 << 24) | (0x01 << 8);
+}
+
+static MXFEssenceElementWriter mxf_bwf_essence_element_writer = {
+ mxf_bwf_get_descriptor,
+ mxf_bwf_update_descriptor,
+ mxf_bwf_get_edit_rate,
+ mxf_bwf_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
+#define BWF_CAPS \
+ "audio/x-raw-int, " \
+ "rate = (int) [ 1, MAX ], " \
+ "channels = (int) [ 1, MAX ], " \
+ "endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, " \
+ "width = (int) 32, " \
+ "depth = (int) 32, " \
+ "signed = (boolean) TRUE; " \
+ "audio/x-raw-int, " \
+ "rate = (int) [ 1, MAX ], " \
+ "channels = (int) [ 1, MAX ], " \
+ "endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, " \
+ "width = (int) 24, " \
+ "depth = (int) 24, " \
+ "signed = (boolean) TRUE; " \
+ "audio/x-raw-int, " \
+ "rate = (int) [ 1, MAX ], " \
+ "channels = (int) [ 1, MAX ], " \
+ "endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, " \
+ "width = (int) 16, " \
+ "depth = (int) 16, " \
+ "signed = (boolean) TRUE; " \
+ "audio/x-raw-int, " \
+ "rate = (int) [ 1, MAX ], " \
+ "channels = (int) [ 1, MAX ], " \
+ "endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, " \
+ "width = (int) 8, " \
+ "depth = (int) 8, " \
+ "signed = (boolean) FALSE"
+
void
mxf_aes_bwf_init (void)
{
@@ -1058,4 +1584,11 @@ mxf_aes_bwf_init (void)
mxf_metadata_register (MXF_TYPE_METADATA_AES3_AUDIO_ESSENCE_DESCRIPTOR);
mxf_essence_element_handler_register (&mxf_aes_bwf_essence_handler);
+
+ mxf_bwf_essence_element_writer.pad_template =
+ gst_pad_template_new ("bwf_audio_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string (BWF_CAPS));
+ memcpy (&mxf_bwf_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_SOUND_ESSENCE), 16);
+ mxf_essence_element_writer_register (&mxf_bwf_essence_element_writer);
}
diff --git a/gst/mxf/mxfalaw.c b/gst/mxf/mxfalaw.c
index 074f1a48..40517af0 100644
--- a/gst/mxf/mxfalaw.c
+++ b/gst/mxf/mxfalaw.c
@@ -29,6 +29,7 @@
#include <string.h>
#include "mxfalaw.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
@@ -138,8 +139,167 @@ static const MXFEssenceElementHandler mxf_alaw_essence_element_handler = {
mxf_alaw_create_caps
};
+typedef struct
+{
+ guint64 error;
+ gint rate, channels;
+ MXFFraction edit_rate;
+} ALawMappingData;
+
+static GstFlowReturn
+mxf_alaw_write_func (GstBuffer * buffer, GstCaps * caps, gpointer mapping_data,
+ GstAdapter * adapter, GstBuffer ** outbuf, gboolean flush)
+{
+ ALawMappingData *md = mapping_data;
+ guint bytes;
+ guint64 speu =
+ gst_util_uint64_scale (md->rate, md->edit_rate.d, md->edit_rate.n);
+
+ md->error += (md->edit_rate.d * md->rate) % (md->edit_rate.n);
+ if (md->error >= md->edit_rate.n) {
+ md->error = 0;
+ speu += 1;
+ }
+
+ bytes = speu * md->channels;
+
+ if (buffer)
+ gst_adapter_push (adapter, buffer);
+
+ if (gst_adapter_available (adapter) == 0)
+ return GST_FLOW_OK;
+
+ if (flush)
+ bytes = MIN (gst_adapter_available (adapter), bytes);
+
+ if (gst_adapter_available (adapter) >= bytes) {
+ *outbuf = gst_adapter_take_buffer (adapter, bytes);
+ }
+
+ if (gst_adapter_available (adapter) >= bytes)
+ return GST_FLOW_CUSTOM_SUCCESS;
+ else
+ return GST_FLOW_OK;
+}
+
+static const guint8 alaw_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x03,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x0a, 0x01, 0x00
+};
+
+static const MXFUL mxf_sound_essence_compression_alaw =
+ { {0x06, 0x0E, 0x2B, 0x34, 0x04, 0x01, 0x01, 0x03, 0x04, 0x02, 0x02, 0x02,
+ 0x03, 0x01, 0x01, 0x00}
+};
+
+static MXFMetadataFileDescriptor *
+mxf_alaw_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataGenericSoundEssenceDescriptor *ret;
+ GstStructure *s;
+ ALawMappingData *md;
+ gint rate, channels;
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "audio/x-alaw") != 0 ||
+ !gst_structure_get_int (s, "rate", &rate) ||
+ !gst_structure_get_int (s, "channels", &channels)) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ return NULL;
+ }
+
+ ret = (MXFMetadataGenericSoundEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_GENERIC_SOUND_ESSENCE_DESCRIPTOR);
+
+ memcpy (&ret->parent.essence_container, &alaw_essence_container_ul, 16);
+ memcpy (&ret->sound_essence_compression, &mxf_sound_essence_compression_alaw,
+ 16);
+
+ if (!mxf_metadata_generic_sound_essence_descriptor_from_caps (ret, caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_alaw_write_func;
+
+ md = g_new0 (ALawMappingData, 1);
+ md->rate = rate;
+ md->channels = channels;
+ *mapping_data = md;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_alaw_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_alaw_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ guint i;
+ gdouble min = G_MAXDOUBLE;
+ ALawMappingData *md = mapping_data;
+
+ for (i = 0; i < package->parent.n_tracks; i++) {
+ MXFMetadataTimelineTrack *tmp;
+
+ if (!MXF_IS_METADATA_TIMELINE_TRACK (package->parent.tracks[i]) ||
+ package->parent.tracks[i] == (MXFMetadataTrack *) track)
+ continue;
+
+ tmp = MXF_METADATA_TIMELINE_TRACK (package->parent.tracks[i]);
+ if (((gdouble) tmp->edit_rate.n) / ((gdouble) tmp->edit_rate.d) < min) {
+ min = ((gdouble) tmp->edit_rate.n) / ((gdouble) tmp->edit_rate.d);
+ memcpy (edit_rate, &tmp->edit_rate, sizeof (MXFFraction));
+ }
+ }
+
+ if (min == G_MAXDOUBLE) {
+ /* 100ms edit units */
+ edit_rate->n = 10;
+ edit_rate->d = 1;
+ }
+
+ memcpy (&md->edit_rate, edit_rate, sizeof (MXFFraction));
+}
+
+static guint32
+mxf_alaw_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x16 << 24) | (0x08 << 8);
+}
+
+static MXFEssenceElementWriter mxf_alaw_essence_element_writer = {
+ mxf_alaw_get_descriptor,
+ mxf_alaw_update_descriptor,
+ mxf_alaw_get_edit_rate,
+ mxf_alaw_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
+#define ALAW_CAPS \
+ "audio/x-alaw, " \
+ "rate = (int) [ 8000, 192000 ], " \
+ "channels = (int) [ 1, 2 ]"
+
void
mxf_alaw_init (void)
{
mxf_essence_element_handler_register (&mxf_alaw_essence_element_handler);
+
+ mxf_alaw_essence_element_writer.pad_template =
+ gst_pad_template_new ("alaw_audio_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string (ALAW_CAPS));
+ memcpy (&mxf_alaw_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_SOUND_ESSENCE), 16);
+ mxf_essence_element_writer_register (&mxf_alaw_essence_element_writer);
}
diff --git a/gst/mxf/mxfdemux.c b/gst/mxf/mxfdemux.c
index 9e486f1e..733e920d 100644
--- a/gst/mxf/mxfdemux.c
+++ b/gst/mxf/mxfdemux.c
@@ -1779,7 +1779,7 @@ gst_mxf_demux_handle_generic_container_essence_element (GstMXFDemux * demux,
pad->last_stop);
if (ret != GST_FLOW_OK)
- break;
+ goto out;
pad->current_essence_track_position++;
diff --git a/gst/mxf/mxfdv-dif.c b/gst/mxf/mxfdv-dif.c
index 10e71b78..220100a3 100644
--- a/gst/mxf/mxfdv-dif.c
+++ b/gst/mxf/mxfdv-dif.c
@@ -24,7 +24,8 @@
/* TODO:
* - playbin hangs on a lot of MXF/DV-DIF files (bug #563827)
* - decodebin2 creates loops inside the linking graph (bug #563828)
- * - Forwarding of timestamps in dvdemux?
+ * - track descriptor might be multiple descriptor, one for sound, one for video
+ * - there might be 2 tracks for one essence, i.e. one audio/one video track
*/
#ifdef HAVE_CONFIG_H
@@ -32,9 +33,11 @@
#endif
#include <gst/gst.h>
+#include <gst/video/video.h>
#include <string.h>
#include "mxfdv-dif.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
@@ -104,6 +107,8 @@ mxf_dv_dif_create_caps (MXFMetadataTimelineTrack * track, GstTagList ** tags,
MXFEssenceElementHandleFunc * handler, gpointer * mapping_data)
{
GstCaps *caps = NULL;
+ guint i;
+ MXFMetadataGenericPictureEssenceDescriptor *d = NULL;
g_return_val_if_fail (track != NULL, NULL);
@@ -112,6 +117,15 @@ mxf_dv_dif_create_caps (MXFMetadataTimelineTrack * track, GstTagList ** tags,
return NULL;
}
+ for (i = 0; i < track->parent.n_descriptor; i++) {
+ if (MXF_IS_METADATA_GENERIC_PICTURE_ESSENCE_DESCRIPTOR (track->
+ parent.descriptor[i])) {
+ d = MXF_METADATA_GENERIC_PICTURE_ESSENCE_DESCRIPTOR (track->
+ parent.descriptor[i]);
+ break;
+ }
+ }
+
*handler = mxf_dv_dif_handle_essence_element;
/* SMPTE 383M 8 */
@@ -123,6 +137,9 @@ mxf_dv_dif_create_caps (MXFMetadataTimelineTrack * track, GstTagList ** tags,
gst_caps_new_simple ("video/x-dv", "systemstream", G_TYPE_BOOLEAN, TRUE,
NULL);
+ if (d)
+ mxf_metadata_generic_picture_essence_descriptor_set_caps (d, caps);
+
if (!*tags)
*tags = gst_tag_list_new ();
@@ -136,8 +153,87 @@ static const MXFEssenceElementHandler mxf_dv_dif_essence_element_handler = {
mxf_dv_dif_create_caps
};
+static GstFlowReturn
+mxf_dv_dif_write_func (GstBuffer * buffer, GstCaps * caps,
+ gpointer mapping_data, GstAdapter * adapter, GstBuffer ** outbuf,
+ gboolean flush)
+{
+ *outbuf = buffer;
+ return GST_FLOW_OK;
+}
+
+static const guint8 dv_dif_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x01,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x02, 0x7f, 0x01
+};
+
+static MXFMetadataFileDescriptor *
+mxf_dv_dif_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataCDCIPictureEssenceDescriptor *ret;
+
+ ret = (MXFMetadataCDCIPictureEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_CDCI_PICTURE_ESSENCE_DESCRIPTOR);
+
+ memcpy (&ret->parent.parent.essence_container, &dv_dif_essence_container_ul,
+ 16);
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+ *handler = mxf_dv_dif_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_dv_dif_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_dv_dif_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ edit_rate->n = a->sample_rate.n;
+ edit_rate->d = a->sample_rate.d;
+}
+
+static guint32
+mxf_dv_dif_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x18 << 24) | (0x01 << 8);
+}
+
+static MXFEssenceElementWriter mxf_dv_dif_essence_element_writer = {
+ mxf_dv_dif_get_descriptor,
+ mxf_dv_dif_update_descriptor,
+ mxf_dv_dif_get_edit_rate,
+ mxf_dv_dif_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
void
mxf_dv_dif_init (void)
{
mxf_essence_element_handler_register (&mxf_dv_dif_essence_element_handler);
+
+ mxf_dv_dif_essence_element_writer.pad_template =
+ gst_pad_template_new ("dv_dif_video_sink_%u", GST_PAD_SINK,
+ GST_PAD_REQUEST,
+ gst_caps_from_string ("video/x-dv, width = "
+ GST_VIDEO_SIZE_RANGE ", height = " GST_VIDEO_SIZE_RANGE
+ ", framerate = " GST_VIDEO_FPS_RANGE ", systemstream = true"));
+ memcpy (&mxf_dv_dif_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_PICTURE_ESSENCE),
+ 16);
+ mxf_essence_element_writer_register (&mxf_dv_dif_essence_element_writer);
}
diff --git a/gst/mxf/mxfjpeg2000.c b/gst/mxf/mxfjpeg2000.c
index c9d7b2aa..9d949d4e 100644
--- a/gst/mxf/mxfjpeg2000.c
+++ b/gst/mxf/mxfjpeg2000.c
@@ -31,9 +31,11 @@
#endif
#include <gst/gst.h>
+#include <gst/video/video.h>
#include <string.h>
#include "mxfjpeg2000.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
@@ -202,8 +204,127 @@ static const MXFEssenceElementHandler mxf_jpeg2000_essence_element_handler = {
mxf_jpeg2000_create_caps
};
+static GstFlowReturn
+mxf_jpeg2000_write_func (GstBuffer * buffer, GstCaps * caps,
+ gpointer mapping_data, GstAdapter * adapter, GstBuffer ** outbuf,
+ gboolean flush)
+{
+ *outbuf = buffer;
+ return GST_FLOW_OK;
+}
+
+static const guint8 jpeg2000_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x07,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x0c, 0x01, 0x00
+};
+
+static const guint jpeg2000_picture_essence_coding[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x04, 0x01, 0x01, 0x07,
+ 0x04, 0x01, 0x02, 0x02, 0x03, 0x01, 0x01, 0x00
+};
+
+static MXFMetadataFileDescriptor *
+mxf_jpeg2000_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataRGBAPictureEssenceDescriptor *ret;
+ GstStructure *s;
+ guint32 fourcc;
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "image/x-jpc") != 0 ||
+ !gst_structure_get_fourcc (s, "fourcc", &fourcc)) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ return NULL;
+ }
+
+ ret = (MXFMetadataRGBAPictureEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_RGBA_PICTURE_ESSENCE_DESCRIPTOR);
+
+ memcpy (&ret->parent.parent.essence_container, &jpeg2000_essence_container_ul,
+ 16);
+ memcpy (&ret->parent.picture_essence_coding, &jpeg2000_picture_essence_coding,
+ 16);
+
+ if (fourcc == GST_MAKE_FOURCC ('s', 'R', 'G', 'B')) {
+ ret->n_pixel_layout = 3;
+ ret->pixel_layout = g_new0 (guint8, 6);
+ ret->pixel_layout[0] = 'R';
+ ret->pixel_layout[1] = 8;
+ ret->pixel_layout[2] = 'G';
+ ret->pixel_layout[3] = 8;
+ ret->pixel_layout[4] = 'B';
+ ret->pixel_layout[5] = 8;
+ } else if (fourcc == GST_MAKE_FOURCC ('s', 'Y', 'U', 'V')) {
+ ret->n_pixel_layout = 3;
+ ret->pixel_layout = g_new0 (guint8, 6);
+ ret->pixel_layout[0] = 'Y';
+ ret->pixel_layout[1] = 8;
+ ret->pixel_layout[2] = 'U';
+ ret->pixel_layout[3] = 8;
+ ret->pixel_layout[4] = 'V';
+ ret->pixel_layout[5] = 8;
+ } else {
+ g_assert_not_reached ();
+ }
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_jpeg2000_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_jpeg2000_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_jpeg2000_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ edit_rate->n = a->sample_rate.n;
+ edit_rate->d = a->sample_rate.d;
+}
+
+static guint32
+mxf_jpeg2000_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x15 << 24) | (0x08 << 8);
+}
+
+static MXFEssenceElementWriter mxf_jpeg2000_essence_element_writer = {
+ mxf_jpeg2000_get_descriptor,
+ mxf_jpeg2000_update_descriptor,
+ mxf_jpeg2000_get_edit_rate,
+ mxf_jpeg2000_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
void
mxf_jpeg2000_init (void)
{
mxf_essence_element_handler_register (&mxf_jpeg2000_essence_element_handler);
+
+ mxf_jpeg2000_essence_element_writer.pad_template =
+ gst_pad_template_new ("jpeg2000_video_sink_%u", GST_PAD_SINK,
+ GST_PAD_REQUEST,
+ gst_caps_from_string ("image/x-jpc, fields = 1, width = "
+ GST_VIDEO_SIZE_RANGE ", height = " GST_VIDEO_SIZE_RANGE
+ ", framerate = " GST_VIDEO_FPS_RANGE
+ ", fourcc = (GstFourcc) { sRGB, sYUV }"));
+ memcpy (&mxf_jpeg2000_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_PICTURE_ESSENCE),
+ 16);
+ mxf_essence_element_writer_register (&mxf_jpeg2000_essence_element_writer);
}
diff --git a/gst/mxf/mxfmetadata.c b/gst/mxf/mxfmetadata.c
index cc53ec15..07c18b0f 100644
--- a/gst/mxf/mxfmetadata.c
+++ b/gst/mxf/mxfmetadata.c
@@ -27,6 +27,7 @@
#include "mxfparse.h"
#include "mxfmetadata.h"
#include "mxfquark.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
@@ -238,6 +239,115 @@ mxf_metadata_base_to_structure (MXFMetadataBase * self)
return NULL;
}
+GstBuffer *
+mxf_metadata_base_to_buffer (MXFMetadataBase * self, MXFPrimerPack * primer)
+{
+ MXFMetadataBaseClass *klass;
+ GstBuffer *ret;
+ GList *tags, *l;
+ guint size = 0, slen;
+ guint8 ber[9];
+ MXFLocalTag *t, *last;
+ guint8 *data;
+
+ g_return_val_if_fail (MXF_IS_METADATA_BASE (self), NULL);
+ g_return_val_if_fail (primer != NULL, NULL);
+
+ klass = MXF_METADATA_BASE_GET_CLASS (self);
+ g_return_val_if_fail (klass->write_tags, NULL);
+
+ tags = klass->write_tags (self, primer);
+ g_return_val_if_fail (tags != NULL, NULL);
+
+ /* Add unknown tags */
+ if (self->other_tags) {
+ MXFLocalTag *tmp;
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ GHashTableIter iter;
+
+ g_hash_table_iter_init (&iter, self->other_tags);
+#else
+ GList *l, *values;
+
+ values = g_hash_table_get_values (self->other_tags);
+#endif
+
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ while (g_hash_table_iter_next (&iter, NULL, (gpointer) & t)) {
+#else
+ for (l = values; l; l = l->next) {
+ t = l->data;
+#endif
+ tmp = g_slice_dup (MXFLocalTag, t);
+ if (t->g_slice) {
+ tmp->data = g_slice_alloc (t->size);
+ mxf_primer_pack_add_mapping (primer, 0x0000, &t->key);
+ memcpy (tmp->data, t->data, t->size);
+ } else {
+ tmp->data = g_memdup (t->data, t->size);
+ }
+ tags = g_list_prepend (tags, tmp);
+ }
+
+#if !GLIB_CHECK_VERSION (2, 16, 0)
+ g_list_free (values);
+#endif
+ }
+
+ l = g_list_last (tags);
+ last = l->data;
+ tags = g_list_delete_link (tags, l);
+ /* Last element contains the metadata UL */
+ g_return_val_if_fail (last->size == 0, NULL);
+
+ for (l = tags; l; l = l->next) {
+ t = l->data;
+ g_assert (G_MAXUINT - t->size >= size);
+ size += 4 + t->size;
+ }
+
+ slen = mxf_ber_encode_size (size, ber);
+ size += 16 + slen;
+
+ ret = gst_buffer_new_and_alloc (size);
+
+ memcpy (GST_BUFFER_DATA (ret), &last->key, 16);
+ mxf_local_tag_free (last);
+ last = NULL;
+ memcpy (GST_BUFFER_DATA (ret) + 16, ber, slen);
+
+ data = GST_BUFFER_DATA (ret) + 16 + slen;
+ size -= 16 + slen;
+
+ for (l = tags; l; l = l->next) {
+ guint16 local_tag;
+
+ g_assert (size >= 4);
+ t = l->data;
+
+ local_tag =
+ GPOINTER_TO_UINT (g_hash_table_lookup (primer->reverse_mappings,
+ &t->key));
+ g_assert (local_tag != 0);
+
+ GST_WRITE_UINT16_BE (data, local_tag);
+ GST_WRITE_UINT16_BE (data + 2, t->size);
+ data += 4;
+ size -= 4;
+ g_assert (size >= t->size);
+
+ memcpy (data, t->data, t->size);
+ data += t->size;
+ size -= t->size;
+
+ mxf_local_tag_free (t);
+ }
+
+ g_list_free (tags);
+
+ return ret;
+}
+
G_DEFINE_ABSTRACT_TYPE (MXFMetadata, mxf_metadata, MXF_TYPE_METADATA_BASE);
static gboolean
@@ -281,12 +391,65 @@ error:
return FALSE;
}
+static GList *
+mxf_metadata_write_tags (MXFMetadataBase * m, MXFPrimerPack * primer)
+{
+ MXFMetadata *self = MXF_METADATA (m);
+ GList *ret = NULL;
+ MXFLocalTag *t;
+ MXFMetadataClass *klass;
+ static const guint8 metadata_key[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x53, 0x01, 0x01,
+ 0x0d, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 instance_uid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x01, 0x01, 0x15, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 generation_uid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+
+ g_return_val_if_fail (MXF_IS_METADATA (self), NULL);
+ klass = MXF_METADATA_GET_CLASS (self);
+
+ /* Last element contains the metadata key */
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, metadata_key, 16);
+ GST_WRITE_UINT16_BE (&t->key.u[13], klass->type);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &instance_uid_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (16);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->parent.instance_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3c0a, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (!mxf_ul_is_zero (&self->parent.generation_uid)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &generation_uid_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (16);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->parent.generation_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x0102, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_class_init (MXFMetadataClass * klass)
{
MXFMetadataBaseClass *metadata_base_class = (MXFMetadataBaseClass *) klass;
metadata_base_class->handle_tag = mxf_metadata_handle_tag;
+ metadata_base_class->write_tags = mxf_metadata_write_tags;
}
static void
@@ -712,6 +875,159 @@ mxf_metadata_preface_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_preface_write_tags (MXFMetadataBase * m, MXFPrimerPack * primer)
+{
+ MXFMetadataPreface *self = MXF_METADATA_PREFACE (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS (mxf_metadata_preface_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ guint i;
+ static const guint8 last_modified_date_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x10, 0x02, 0x04, 0x00, 0x00
+ };
+ static const guint8 version_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x03, 0x01, 0x02, 0x01, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 object_model_version_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x03, 0x01, 0x02, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 primary_package_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x06, 0x01, 0x01, 0x04, 0x01, 0x08, 0x00, 0x00
+ };
+ static const guint8 identifications_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x06, 0x04, 0x00, 0x00
+ };
+ static const guint8 content_storage_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x02, 0x01, 0x00, 0x00
+ };
+ static const guint8 operational_pattern_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x01, 0x02, 0x02, 0x03, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 essence_containers_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x01, 0x02, 0x02, 0x10, 0x02, 0x01, 0x00, 0x00
+ };
+ static const guint8 dm_schemes_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x01, 0x02, 0x02, 0x10, 0x02, 0x02, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &last_modified_date_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (8);
+ t->g_slice = TRUE;
+ mxf_timestamp_write (&self->last_modified_date, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x3b02, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &version_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (2);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->version);
+ mxf_primer_pack_add_mapping (primer, 0x3b05, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->object_model_version) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &object_model_version_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (4);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->object_model_version);
+ mxf_primer_pack_add_mapping (primer, 0x3b07, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_ul_is_zero (&self->primary_package_uid)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &primary_package_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (16);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->primary_package_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3b08, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &identifications_ul, 16);
+ t->size = 8 + 16 * self->n_identifications;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ mxf_primer_pack_add_mapping (primer, 0x3b06, &t->key);
+ GST_WRITE_UINT32_BE (t->data, self->n_identifications);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_identifications; i++) {
+ if (!self->identifications[i])
+ continue;
+
+ memcpy (t->data + 8 + 16 * i,
+ &MXF_METADATA_BASE (self->identifications[i])->instance_uid, 16);
+ }
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &content_storage_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_primer_pack_add_mapping (primer, 0x3b03, &t->key);
+ memcpy (t->data, &MXF_METADATA_BASE (self->content_storage)->instance_uid,
+ 16);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &operational_pattern_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_primer_pack_add_mapping (primer, 0x3b09, &t->key);
+ memcpy (t->data, &self->operational_pattern, 16);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &essence_containers_ul, 16);
+ t->size = 8 + 16 * self->n_essence_containers;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ mxf_primer_pack_add_mapping (primer, 0x3b0a, &t->key);
+ GST_WRITE_UINT32_BE (t->data, self->n_essence_containers);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_essence_containers; i++) {
+ memcpy (t->data + 8 + 16 * i, &self->essence_containers[i], 16);
+ }
+ ret = g_list_prepend (ret, t);
+
+ if (self->dm_schemes) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &dm_schemes_ul, 16);
+ t->size = 8 + 16 * self->n_dm_schemes;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ mxf_primer_pack_add_mapping (primer, 0x3b0b, &t->key);
+ GST_WRITE_UINT32_BE (t->data, self->n_dm_schemes);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_dm_schemes; i++) {
+ memcpy (t->data + 8 + 16 * i, &self->dm_schemes[i], 16);
+ }
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_preface_init (MXFMetadataPreface * self)
{
@@ -729,6 +1045,7 @@ mxf_metadata_preface_class_init (MXFMetadataPrefaceClass * klass)
metadata_base_class->handle_tag = mxf_metadata_preface_handle_tag;
metadata_base_class->resolve = mxf_metadata_preface_resolve;
metadata_base_class->to_structure = mxf_metadata_preface_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_preface_write_tags;
metadata_base_class->name_quark = MXF_QUARK (PREFACE);
metadata_class->type = 0x012f;
}
@@ -912,6 +1229,127 @@ mxf_metadata_identification_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_identification_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataIdentification *self = MXF_METADATA_IDENTIFICATION (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_identification_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 company_name_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x02, 0x01, 0x00, 0x00
+ };
+ static const guint8 product_name_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x03, 0x01, 0x00, 0x00
+ };
+ static const guint8 product_version_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 version_string_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x05, 0x01, 0x00, 0x00
+ };
+ static const guint8 product_uid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x07, 0x00, 0x00, 0x00
+ };
+ static const guint8 modification_date_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x10, 0x02, 0x03, 0x00, 0x00
+ };
+ static const guint8 toolkit_version_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x0A, 0x00, 0x00, 0x00
+ };
+ static const guint8 platform_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x07, 0x01, 0x06, 0x01, 0x00, 0x00
+ };
+
+ if (self->company_name) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &company_name_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->company_name, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3c01, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->product_name) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &product_name_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->product_name, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3c02, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_product_version_is_valid (&self->product_version)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &product_version_ul, 16);
+ t->size = 10;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_product_version_write (&self->product_version, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x3c03, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->version_string) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &version_string_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->version_string, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3c04, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_ul_is_zero (&self->product_uid)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &product_uid_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->product_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3c05, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_timestamp_is_unknown (&self->modification_date)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &modification_date_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_timestamp_write (&self->modification_date, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x3c06, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_product_version_is_valid (&self->toolkit_version)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &toolkit_version_ul, 16);
+ t->size = 10;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_product_version_write (&self->toolkit_version, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x3c07, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->platform) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &platform_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->platform, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x3c08, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_identification_init (MXFMetadataIdentification * self)
{
@@ -929,6 +1367,7 @@ mxf_metadata_identification_class_init (MXFMetadataIdentificationClass * klass)
metadata_base_class->handle_tag = mxf_metadata_identification_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (IDENTIFICATION);
metadata_base_class->to_structure = mxf_metadata_identification_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_identification_write_tags;
metadata_class->type = 0x0130;
}
@@ -1151,6 +1590,69 @@ mxf_metadata_content_storage_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_content_storage_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataContentStorage *self = MXF_METADATA_CONTENT_STORAGE (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_content_storage_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ guint i;
+ static const guint8 packages_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x05, 0x01, 0x00, 0x00
+ };
+ static const guint8 essence_container_data_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x05, 0x02, 0x00, 0x00
+ };
+
+ if (self->packages) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &packages_ul, 16);
+ t->size = 8 + 16 * self->n_packages;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_packages);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_packages; i++) {
+ if (!self->packages[i])
+ continue;
+
+ memcpy (t->data + 8 + i * 16,
+ &MXF_METADATA_BASE (self->packages[i])->instance_uid, 16);
+ }
+
+ mxf_primer_pack_add_mapping (primer, 0x1901, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->essence_container_data) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &essence_container_data_ul, 16);
+ t->size = 8 + 16 * self->n_essence_container_data;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_essence_container_data);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_essence_container_data; i++) {
+ if (!self->essence_container_data[i])
+ continue;
+
+ memcpy (t->data + 8 + i * 16,
+ &MXF_METADATA_BASE (self->essence_container_data[i])->instance_uid,
+ 16);
+ }
+
+ mxf_primer_pack_add_mapping (primer, 0x1902, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_content_storage_init (MXFMetadataContentStorage * self)
{
@@ -1169,6 +1671,7 @@ mxf_metadata_content_storage_class_init (MXFMetadataContentStorageClass * klass)
metadata_base_class->resolve = mxf_metadata_content_storage_resolve;
metadata_base_class->name_quark = MXF_QUARK (CONTENT_STORAGE);
metadata_base_class->to_structure = mxf_metadata_content_storage_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_content_storage_write_tags;
metadata_class->type = 0x0118;
}
@@ -1298,6 +1801,63 @@ mxf_metadata_essence_container_data_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_essence_container_data_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataEssenceContainerData *self =
+ MXF_METADATA_ESSENCE_CONTAINER_DATA (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_essence_container_data_parent_class)->write_tags (m,
+ primer);
+ MXFLocalTag *t;
+ static const guint8 linked_package_uid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x06, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 body_sid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x01, 0x03, 0x04, 0x04, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 index_sid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x01, 0x03, 0x04, 0x05, 0x00, 0x00, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &linked_package_uid_ul, 16);
+ t->size = 32;
+ t->data = g_slice_alloc0 (32);
+ t->g_slice = TRUE;
+ if (self->linked_package)
+ memcpy (t->data, &self->linked_package->parent.package_uid, 32);
+ mxf_primer_pack_add_mapping (primer, 0x2701, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &body_sid_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (4);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->body_sid);
+ mxf_primer_pack_add_mapping (primer, 0x3f07, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->index_sid) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &index_sid_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (4);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->index_sid);
+ mxf_primer_pack_add_mapping (primer, 0x3f07, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_essence_container_data_init (MXFMetadataEssenceContainerData *
self)
@@ -1318,6 +1878,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (ESSENCE_CONTAINER_DATA);
metadata_base_class->to_structure =
mxf_metadata_essence_container_data_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_essence_container_data_write_tags;
metadata_class->type = 0x0123;
}
@@ -1518,6 +2080,95 @@ mxf_metadata_generic_package_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_generic_package_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataGenericPackage *self = MXF_METADATA_GENERIC_PACKAGE (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_generic_package_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 package_uid_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x01, 0x01, 0x15, 0x10, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 name_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x01, 0x03, 0x03, 0x02, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 package_creation_date_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x10, 0x01, 0x03, 0x00, 0x00
+ };
+ static const guint8 package_modified_date_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x10, 0x02, 0x05, 0x00, 0x00
+ };
+ static const guint8 tracks_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x06, 0x05, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &package_uid_ul, 16);
+ t->size = 32;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->package_uid, 32);
+ mxf_primer_pack_add_mapping (primer, 0x4401, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->name) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &name_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->name, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x4402, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &package_creation_date_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_timestamp_write (&self->package_creation_date, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x4405, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &package_modified_date_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ mxf_timestamp_write (&self->package_modified_date, t->data);
+ mxf_primer_pack_add_mapping (primer, 0x4404, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->tracks) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &tracks_ul, 16);
+ t->size = 8 + 16 * self->n_tracks;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_tracks);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_tracks; i++) {
+ if (!self->tracks[i])
+ continue;
+
+ memcpy (t->data + 8 + 16 * i,
+ &MXF_METADATA_BASE (self->tracks[i])->instance_uid, 16);
+ }
+ mxf_primer_pack_add_mapping (primer, 0x4403, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_generic_package_init (MXFMetadataGenericPackage * self)
{
@@ -1534,6 +2185,7 @@ mxf_metadata_generic_package_class_init (MXFMetadataGenericPackageClass * klass)
metadata_base_class->handle_tag = mxf_metadata_generic_package_handle_tag;
metadata_base_class->resolve = mxf_metadata_generic_package_resolve;
metadata_base_class->to_structure = mxf_metadata_generic_package_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_generic_package_write_tags;
}
G_DEFINE_TYPE (MXFMetadataMaterialPackage, mxf_metadata_material_package,
@@ -1802,6 +2454,34 @@ mxf_metadata_source_package_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_source_package_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataSourcePackage *self = MXF_METADATA_SOURCE_PACKAGE (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_source_package_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 descriptor_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x02, 0x03, 0x00, 0x00
+ };
+
+ if (self->descriptor) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &descriptor_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &MXF_METADATA_BASE (self->descriptor)->instance_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x4701, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_source_package_init (MXFMetadataSourcePackage * self)
{
@@ -1818,6 +2498,7 @@ mxf_metadata_source_package_class_init (MXFMetadataSourcePackageClass * klass)
metadata_base_class->resolve = mxf_metadata_source_package_resolve;
metadata_base_class->name_quark = MXF_QUARK (SOURCE_PACKAGE);
metadata_base_class->to_structure = mxf_metadata_source_package_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_source_package_write_tags;
metadata_class->type = 0x0137;
}
@@ -1959,6 +2640,69 @@ mxf_metadata_track_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_track_write_tags (MXFMetadataBase * m, MXFPrimerPack * primer)
+{
+ MXFMetadataTrack *self = MXF_METADATA_TRACK (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS (mxf_metadata_track_parent_class)->write_tags (m,
+ primer);
+ MXFLocalTag *t;
+ static const guint8 track_id_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x01, 0x07, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 track_number_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x01, 0x04, 0x01, 0x03, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 track_name_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x01, 0x07, 0x01, 0x02, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 sequence_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x02, 0x04, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &track_id_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->track_id);
+ mxf_primer_pack_add_mapping (primer, 0x4801, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &track_number_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->track_number);
+ mxf_primer_pack_add_mapping (primer, 0x4804, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->track_name) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &track_name_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->track_name, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x4802, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sequence_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &MXF_METADATA_BASE (self->sequence)->instance_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x4803, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
+
static void
mxf_metadata_track_init (MXFMetadataTrack * self)
{
@@ -1975,6 +2719,7 @@ mxf_metadata_track_class_init (MXFMetadataTrackClass * klass)
metadata_base_class->handle_tag = mxf_metadata_track_handle_tag;
metadata_base_class->resolve = mxf_metadata_track_resolve;
metadata_base_class->to_structure = mxf_metadata_track_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_track_write_tags;
}
/* SMPTE RP224 */
@@ -2022,6 +2767,18 @@ mxf_metadata_track_identifier_parse (const MXFUL * track_identifier)
return MXF_METADATA_TRACK_UNKNOWN;
}
+const MXFUL *
+mxf_metadata_track_identifier_get (MXFMetadataTrackType type)
+{
+ guint i;
+
+ for (i = 0; i < G_N_ELEMENTS (mxf_metadata_track_identifier); i++)
+ if (mxf_metadata_track_identifier[i].type == type)
+ return (const MXFUL *) &mxf_metadata_track_identifier[i].ul;
+
+ return NULL;
+}
+
G_DEFINE_TYPE (MXFMetadataTimelineTrack, mxf_metadata_timeline_track,
MXF_TYPE_METADATA_TRACK);
@@ -2078,6 +2835,46 @@ mxf_metadata_timeline_track_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_timeline_track_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataTimelineTrack *self = MXF_METADATA_TIMELINE_TRACK (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_timeline_track_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 edit_rate_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x30, 0x04, 0x05, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 origin_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x03, 0x01, 0x03, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &edit_rate_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->edit_rate.n);
+ GST_WRITE_UINT32_BE (t->data + 4, self->edit_rate.d);
+ mxf_primer_pack_add_mapping (primer, 0x4b01, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &origin_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->origin);
+ mxf_primer_pack_add_mapping (primer, 0x4b02, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
+
static void
mxf_metadata_timeline_track_init (MXFMetadataTimelineTrack * self)
{
@@ -2093,6 +2890,7 @@ mxf_metadata_timeline_track_class_init (MXFMetadataTimelineTrackClass * klass)
metadata_base_class->handle_tag = mxf_metadata_timeline_track_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (TIMELINE_TRACK);
metadata_base_class->to_structure = mxf_metadata_timeline_track_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_timeline_track_write_tags;
metadata_class->type = 0x013b;
}
@@ -2152,6 +2950,46 @@ mxf_metadata_event_track_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_event_track_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataEventTrack *self = MXF_METADATA_EVENT_TRACK (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_event_track_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 event_edit_rate_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x30, 0x04, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 event_origin_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x07, 0x02, 0x01, 0x03, 0x01, 0x0B, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &event_edit_rate_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->event_edit_rate.n);
+ GST_WRITE_UINT32_BE (t->data + 4, self->event_edit_rate.d);
+ mxf_primer_pack_add_mapping (primer, 0x4901, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &event_origin_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->event_origin);
+ mxf_primer_pack_add_mapping (primer, 0x4902, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
+
static void
mxf_metadata_event_track_init (MXFMetadataEventTrack * self)
{
@@ -2167,6 +3005,7 @@ mxf_metadata_event_track_class_init (MXFMetadataEventTrackClass * klass)
metadata_base_class->handle_tag = mxf_metadata_event_track_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (EVENT_TRACK);
metadata_base_class->to_structure = mxf_metadata_event_track_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_event_track_write_tags;
metadata_class->type = 0x0139;
}
@@ -2345,6 +3184,71 @@ mxf_metadata_sequence_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_sequence_write_tags (MXFMetadataBase * m, MXFPrimerPack * primer)
+{
+ MXFMetadataSequence *self = MXF_METADATA_SEQUENCE (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS (mxf_metadata_sequence_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 data_definition_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x07, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 duration_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x02, 0x01, 0x01, 0x03, 0x00, 0x00
+ };
+ static const guint8 structural_components_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x06, 0x09, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &data_definition_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->data_definition, 16);
+ mxf_primer_pack_add_mapping (primer, 0x0201, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &duration_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->duration);
+ mxf_primer_pack_add_mapping (primer, 0x0202, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->structural_components) {
+ guint i;
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &structural_components_ul, 16);
+ t->size = 8 + 16 * self->n_structural_components;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+
+ GST_WRITE_UINT32_BE (t->data, self->n_structural_components);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_structural_components; i++) {
+ if (!self->structural_components[i])
+ continue;
+
+ memcpy (t->data + 8 + i * 16,
+ &MXF_METADATA_BASE (self->structural_components[i])->instance_uid,
+ 16);
+ }
+
+ mxf_primer_pack_add_mapping (primer, 0x1001, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_sequence_init (MXFMetadataSequence * self)
{
@@ -2363,6 +3267,7 @@ mxf_metadata_sequence_class_init (MXFMetadataSequenceClass * klass)
metadata_base_class->resolve = mxf_metadata_sequence_resolve;
metadata_base_class->name_quark = MXF_QUARK (SEQUENCE);
metadata_base_class->to_structure = mxf_metadata_sequence_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_sequence_write_tags;
metadata_class->type = 0x010f;
}
@@ -2429,6 +3334,45 @@ mxf_metadata_structural_component_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_structural_component_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataStructuralComponent *self = MXF_METADATA_STRUCTURAL_COMPONENT (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_structural_component_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 data_definition_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x07, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 duration_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x02, 0x01, 0x01, 0x03, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &data_definition_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->data_definition, 16);
+ mxf_primer_pack_add_mapping (primer, 0x0201, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &duration_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->duration);
+ mxf_primer_pack_add_mapping (primer, 0x0202, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
+
static void
mxf_metadata_structural_component_init (MXFMetadataStructuralComponent * self)
{
@@ -2445,6 +3389,8 @@ static void
mxf_metadata_structural_component_handle_tag;
metadata_base_class->to_structure =
mxf_metadata_structural_component_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_structural_component_write_tags;
}
G_DEFINE_TYPE (MXFMetadataTimecodeComponent, mxf_metadata_timecode_component,
@@ -2512,6 +3458,58 @@ mxf_metadata_timecode_component_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_timecode_component_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataTimecodeComponent *self = MXF_METADATA_TIMECODE_COMPONENT (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_timecode_component_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 rounded_timecode_base_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x04, 0x01, 0x01, 0x02, 0x06, 0x00, 0x00
+ };
+ static const guint8 start_timecode_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x03, 0x01, 0x05, 0x00, 0x00
+ };
+ static const guint8 drop_frame_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x04, 0x01, 0x01, 0x05, 0x00, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &rounded_timecode_base_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->rounded_timecode_base);
+ mxf_primer_pack_add_mapping (primer, 0x1502, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &start_timecode_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->start_timecode);
+ mxf_primer_pack_add_mapping (primer, 0x1501, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &drop_frame_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->drop_frame) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0x1503, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
+
static void
mxf_metadata_timecode_component_init (MXFMetadataTimecodeComponent * self)
{
@@ -2529,6 +3527,7 @@ mxf_metadata_timecode_component_class_init (MXFMetadataTimecodeComponentClass *
metadata_base_class->name_quark = MXF_QUARK (TIMECODE_COMPONENT);
metadata_base_class->to_structure =
mxf_metadata_timecode_component_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_timecode_component_write_tags;
metadata_class->type = 0x0114;
}
@@ -2643,6 +3642,57 @@ mxf_metadata_source_clip_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_source_clip_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataSourceClip *self = MXF_METADATA_SOURCE_CLIP (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_source_clip_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 start_position_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x03, 0x01, 0x04, 0x00, 0x00
+ };
+ static const guint8 source_package_id_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x03, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 source_track_id_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x03, 0x02, 0x00, 0x00, 0x00
+ };
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &start_position_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->start_position);
+ mxf_primer_pack_add_mapping (primer, 0x1201, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &source_package_id_ul, 16);
+ t->size = 32;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->source_package_id, 32);
+ mxf_primer_pack_add_mapping (primer, 0x1101, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &source_track_id_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->source_track_id);
+ mxf_primer_pack_add_mapping (primer, 0x1102, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ return ret;
+}
static void
mxf_metadata_source_clip_init (MXFMetadataSourceClip * self)
@@ -2660,6 +3710,7 @@ mxf_metadata_source_clip_class_init (MXFMetadataSourceClipClass * klass)
metadata_base_class->resolve = mxf_metadata_source_clip_resolve;
metadata_base_class->name_quark = MXF_QUARK (SOURCE_CLIP);
metadata_base_class->to_structure = mxf_metadata_source_clip_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_source_clip_write_tags;
metadata_class->type = 0x0111;
}
@@ -2771,6 +3822,39 @@ mxf_metadata_dm_source_clip_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_dm_source_clip_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataDMSourceClip *self = MXF_METADATA_DM_SOURCE_CLIP (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_dm_source_clip_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 track_ids_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x01, 0x07, 0x01, 0x06, 0x00, 0x00, 0x00, 0x00
+ };
+
+ if (self->track_ids) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &track_ids_ul, 16);
+ t->size = 8 + 4 * self->n_track_ids;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_track_ids);
+ GST_WRITE_UINT32_BE (t->data + 4, 4);
+ for (i = 0; i < self->n_track_ids; i++)
+ GST_WRITE_UINT32_BE (t->data + 8 + i * 4, self->track_ids[i]);
+ mxf_primer_pack_add_mapping (primer, 0x6103, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_dm_source_clip_init (MXFMetadataDMSourceClip * self)
{
@@ -2788,6 +3872,7 @@ mxf_metadata_dm_source_clip_class_init (MXFMetadataDMSourceClipClass * klass)
metadata_base_class->handle_tag = mxf_metadata_dm_source_clip_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (DM_SOURCE_CLIP);
metadata_base_class->to_structure = mxf_metadata_dm_source_clip_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_dm_source_clip_write_tags;
metadata_class->type = 0x0145;
}
@@ -2965,10 +4050,84 @@ mxf_metadata_dm_segment_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_dm_segment_write_tags (MXFMetadataBase * m, MXFPrimerPack * primer)
+{
+ MXFMetadataDMSegment *self = MXF_METADATA_DM_SEGMENT (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS (mxf_metadata_dm_segment_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 event_start_position_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x07, 0x02, 0x01, 0x03, 0x03, 0x03, 0x00, 0x00
+ };
+ static const guint8 event_comment_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x30, 0x04, 0x04, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 track_ids_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x01, 0x07, 0x01, 0x05, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 dm_framework_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x06, 0x01, 0x01, 0x04, 0x02, 0x0C, 0x00, 0x00
+ };
+
+ if (self->event_start_position != -1) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &event_start_position_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->event_start_position);
+ mxf_primer_pack_add_mapping (primer, 0x0601, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->event_comment) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &event_comment_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->event_comment, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x0602, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->track_ids) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &track_ids_ul, 16);
+ t->size = 8 + 4 * self->n_track_ids;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_track_ids);
+ GST_WRITE_UINT32_BE (t->data + 4, 4);
+ for (i = 0; i < self->n_track_ids; i++)
+ GST_WRITE_UINT32_BE (t->data + 8 + i * 4, self->track_ids[i]);
+ mxf_primer_pack_add_mapping (primer, 0x6102, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->dm_framework) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &dm_framework_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &MXF_METADATA_BASE (self->dm_framework)->instance_uid, 16);
+ mxf_primer_pack_add_mapping (primer, 0x6101, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_dm_segment_init (MXFMetadataDMSegment * self)
{
-
+ self->event_start_position = -1;
}
static void
@@ -2983,6 +4142,7 @@ mxf_metadata_dm_segment_class_init (MXFMetadataDMSegmentClass * klass)
metadata_base_class->resolve = mxf_metadata_dm_segment_resolve;
metadata_base_class->name_quark = MXF_QUARK (DM_SEGMENT);
metadata_base_class->to_structure = mxf_metadata_dm_segment_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_dm_segment_write_tags;
metadata_class->type = 0x0141;
}
@@ -3129,6 +4289,43 @@ mxf_metadata_generic_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_generic_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataGenericDescriptor *self = MXF_METADATA_GENERIC_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_generic_descriptor_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 locators_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x06, 0x03, 0x00, 0x00
+ };
+
+ if (self->locators) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &locators_ul, 16);
+ t->size = 8 + 16 * self->n_locators;;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_locators);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_locators; i++) {
+ if (!self->locators[i])
+ continue;
+ memcpy (t->data + 8 + 16 * i,
+ &MXF_METADATA_BASE (self->locators[i])->instance_uid, 16);
+ }
+ mxf_primer_pack_add_mapping (primer, 0x2f01, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_generic_descriptor_init (MXFMetadataGenericDescriptor * self)
{
@@ -3147,6 +4344,7 @@ mxf_metadata_generic_descriptor_class_init (MXFMetadataGenericDescriptorClass *
metadata_base_class->resolve = mxf_metadata_generic_descriptor_resolve;
metadata_base_class->to_structure =
mxf_metadata_generic_descriptor_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_generic_descriptor_write_tags;
}
G_DEFINE_TYPE (MXFMetadataFileDescriptor, mxf_metadata_file_descriptor,
@@ -3247,6 +4445,91 @@ mxf_metadata_file_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_file_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataFileDescriptor *self = MXF_METADATA_FILE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_file_descriptor_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 linked_track_id_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x06, 0x01, 0x01, 0x03, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 sample_rate_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x06, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 container_duration_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x06, 0x01, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 essence_container_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x01, 0x02, 0x00, 0x00
+ };
+ static const guint8 codec_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x06, 0x01, 0x01, 0x04, 0x01, 0x03, 0x00, 0x00
+ };
+
+ if (self->linked_track_id) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &linked_track_id_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->linked_track_id);
+ mxf_primer_pack_add_mapping (primer, 0x3006, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sample_rate_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->sample_rate.n);
+ GST_WRITE_UINT32_BE (t->data + 4, self->sample_rate.d);
+ mxf_primer_pack_add_mapping (primer, 0x3001, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->container_duration > 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &container_duration_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->container_duration);
+ mxf_primer_pack_add_mapping (primer, 0x3002, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &essence_container_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->essence_container, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3004, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (!mxf_ul_is_zero (&self->codec)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &codec_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->codec, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3005, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_file_descriptor_init (MXFMetadataFileDescriptor * self)
{
@@ -3262,6 +4545,7 @@ mxf_metadata_file_descriptor_class_init (MXFMetadataFileDescriptorClass * klass)
metadata_base_class->handle_tag = mxf_metadata_file_descriptor_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (FILE_DESCRIPTOR);
metadata_base_class->to_structure = mxf_metadata_file_descriptor_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_file_descriptor_write_tags;
metadata_class->type = 0x0125;
}
@@ -3567,6 +4851,383 @@ mxf_metadata_generic_picture_essence_descriptor_to_structure (MXFMetadataBase *
return ret;
}
+static GList *
+mxf_metadata_generic_picture_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataGenericPictureEssenceDescriptor *self =
+ MXF_METADATA_GENERIC_PICTURE_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_generic_picture_essence_descriptor_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 signal_standard_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x05, 0x01, 0x13, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 frame_layout_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x03, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 stored_width_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x02, 0x02, 0x00, 0x00, 0x00
+ };
+ static const guint8 stored_height_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x02, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 stored_f2_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x03, 0x02, 0x08, 0x00, 0x00, 0x00
+ };
+ static const guint8 sampled_width_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x08, 0x00, 0x00, 0x00
+ };
+ static const guint8 sampled_height_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x07, 0x00, 0x00, 0x00
+ };
+ static const guint8 sampled_x_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x09, 0x00, 0x00, 0x00
+ };
+ static const guint8 sampled_y_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x0A, 0x00, 0x00, 0x00
+ };
+ static const guint8 display_height_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x0B, 0x00, 0x00, 0x00
+ };
+ static const guint8 display_width_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x0C, 0x00, 0x00, 0x00
+ };
+ static const guint8 display_x_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x0D, 0x00, 0x00, 0x00
+ };
+ static const guint8 display_y_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x0E, 0x00, 0x00, 0x00
+ };
+ static const guint8 display_f2_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x03, 0x02, 0x07, 0x00, 0x00, 0x00
+ };
+ static const guint8 aspect_ratio_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 active_format_descriptor_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x03, 0x02, 0x09, 0x00, 0x00, 0x00
+ };
+ static const guint8 video_line_map_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x03, 0x02, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 alpha_transparency_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x05, 0x20, 0x01, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 capture_gamma_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x02, 0x01, 0x01, 0x01, 0x02, 0x00
+ };
+ static const guint8 image_alignment_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x18, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 image_start_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x18, 0x02, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 image_end_offset_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x18, 0x03, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 field_dominance_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x03, 0x01, 0x06, 0x00, 0x00, 0x00
+ };
+ static const guint8 picture_essence_coding_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x06, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+
+ if (self->signal_standard != 1) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &signal_standard_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->signal_standard);
+ mxf_primer_pack_add_mapping (primer, 0x3215, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->frame_layout != 255) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &frame_layout_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->frame_layout);
+ mxf_primer_pack_add_mapping (primer, 0x320c, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->stored_width != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &stored_width_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->stored_width);
+ mxf_primer_pack_add_mapping (primer, 0x3203, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->stored_height != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &stored_height_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->stored_height);
+ mxf_primer_pack_add_mapping (primer, 0x3202, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->stored_f2_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &stored_f2_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->stored_f2_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3216, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->sampled_width != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sampled_width_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->sampled_width);
+ mxf_primer_pack_add_mapping (primer, 0x3205, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->sampled_height != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sampled_height_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->sampled_height);
+ mxf_primer_pack_add_mapping (primer, 0x3204, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->sampled_x_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sampled_x_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->sampled_x_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3206, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->sampled_y_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sampled_y_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->sampled_y_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3207, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->display_height != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &display_height_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->display_height);
+ mxf_primer_pack_add_mapping (primer, 0x3208, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->display_width != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &display_width_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->display_width);
+ mxf_primer_pack_add_mapping (primer, 0x3209, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->display_x_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &display_x_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->display_x_offset);
+ mxf_primer_pack_add_mapping (primer, 0x320a, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->display_y_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &display_y_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->display_y_offset);
+ mxf_primer_pack_add_mapping (primer, 0x320b, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->display_f2_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &display_f2_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->display_f2_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3217, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->aspect_ratio.n != 0 && self->aspect_ratio.d != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &aspect_ratio_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->aspect_ratio.n);
+ GST_WRITE_UINT32_BE (t->data + 4, self->aspect_ratio.d);
+ mxf_primer_pack_add_mapping (primer, 0x320e, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->active_format_descriptor != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &active_format_descriptor_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->active_format_descriptor);
+ mxf_primer_pack_add_mapping (primer, 0x3218, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->video_line_map[0] != 0 || self->video_line_map[1] != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &video_line_map_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT64_BE (t->data, self->video_line_map[0]);
+ GST_WRITE_UINT64_BE (t->data + 8, self->video_line_map[1]);
+ mxf_primer_pack_add_mapping (primer, 0x320d, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->alpha_transparency != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &alpha_transparency_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->alpha_transparency);
+ mxf_primer_pack_add_mapping (primer, 0x320f, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_ul_is_zero (&self->capture_gamma)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &capture_gamma_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->capture_gamma, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3210, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->image_alignment_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &image_alignment_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->image_alignment_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3211, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->image_start_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &image_start_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->image_start_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3213, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->image_end_offset != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &image_end_offset_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->image_end_offset);
+ mxf_primer_pack_add_mapping (primer, 0x3214, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->field_dominance != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &field_dominance_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->field_dominance);
+ mxf_primer_pack_add_mapping (primer, 0x3212, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_ul_is_zero (&self->picture_essence_coding)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &picture_essence_coding_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->picture_essence_coding, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3201, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_generic_picture_essence_descriptor_init
(MXFMetadataGenericPictureEssenceDescriptor * self)
@@ -3588,6 +5249,8 @@ static void
MXF_QUARK (GENERIC_PICTURE_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_generic_picture_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_generic_picture_essence_descriptor_write_tags;
metadata_class->type = 0x0127;
}
@@ -3641,6 +5304,66 @@ void mxf_metadata_generic_picture_essence_descriptor_set_caps
par_n, par_d, NULL);
}
+static gint
+gst_greatest_common_divisor (gint a, gint b)
+{
+ while (b != 0) {
+ int temp = a;
+
+ a = b;
+ b = temp % b;
+ }
+
+ return ABS (a);
+}
+
+gboolean
+ mxf_metadata_generic_picture_essence_descriptor_from_caps
+ (MXFMetadataGenericPictureEssenceDescriptor * self, GstCaps * caps) {
+ gint par_n, par_d, gcd;
+ gint width, height;
+ gint fps_n, fps_d;
+ MXFMetadataFileDescriptor *f = (MXFMetadataFileDescriptor *) self;
+ GstStructure *s;
+
+ g_return_val_if_fail (MXF_IS_METADATA_GENERIC_PICTURE_ESSENCE_DESCRIPTOR
+ (self), FALSE);
+ g_return_val_if_fail (GST_IS_CAPS (caps), FALSE);
+
+ s = gst_caps_get_structure (caps, 0);
+
+ if (!gst_structure_get_fraction (s, "framerate", &fps_n, &fps_d)) {
+ GST_ERROR ("Invalid framerate");
+ return FALSE;
+ }
+ f->sample_rate.n = fps_n;
+ f->sample_rate.d = fps_d;
+
+ if (!gst_structure_get_int (s, "width", &width) ||
+ !gst_structure_get_int (s, "height", &height)) {
+ GST_ERROR ("Invalid width/height");
+ return FALSE;
+ }
+
+ self->stored_width = width;
+ self->stored_height = height;
+
+ if (!gst_structure_get_fraction (s, "pixel-aspect-ratio", &par_n, &par_d)) {
+ par_n = 1;
+ par_d = 1;
+ }
+
+ self->aspect_ratio.n = par_n * width;
+ self->aspect_ratio.d = par_d * height;
+ gcd =
+ gst_greatest_common_divisor (self->aspect_ratio.n, self->aspect_ratio.d);
+ self->aspect_ratio.n /= gcd;
+ self->aspect_ratio.d /= gcd;
+
+ return TRUE;
+}
+
+
G_DEFINE_TYPE (MXFMetadataGenericSoundEssenceDescriptor,
mxf_metadata_generic_sound_essence_descriptor,
MXF_TYPE_METADATA_FILE_DESCRIPTOR);
@@ -3770,12 +5493,147 @@ mxf_metadata_generic_sound_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_generic_sound_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataGenericSoundEssenceDescriptor *self =
+ MXF_METADATA_GENERIC_SOUND_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_generic_sound_essence_descriptor_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 audio_sampling_rate_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00
+ };
+ static const guint8 locked_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x04, 0x02, 0x03, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 audio_ref_level_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x02, 0x01, 0x01, 0x03, 0x00, 0x00, 0x00
+ };
+ static const guint8 electro_spatial_formulation_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 channel_count_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x01, 0x01, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 quantization_bits_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x04, 0x02, 0x03, 0x03, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 dial_norm_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x02, 0x07, 0x01, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 sound_essence_compression_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x02, 0x04, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+
+ if (self->audio_sampling_rate.d && self->audio_sampling_rate.n) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &audio_sampling_rate_ul, 16);
+ t->size = 8;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->audio_sampling_rate.n);
+ GST_WRITE_UINT32_BE (t->data + 4, self->audio_sampling_rate.d);
+ mxf_primer_pack_add_mapping (primer, 0x3d03, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &locked_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->locked) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0x3d02, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ if (self->audio_ref_level) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &audio_ref_level_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->audio_ref_level);
+ mxf_primer_pack_add_mapping (primer, 0x3d04, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->electro_spatial_formulation != 255) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &electro_spatial_formulation_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->electro_spatial_formulation);
+ mxf_primer_pack_add_mapping (primer, 0x3d05, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->channel_count) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &channel_count_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->channel_count);
+ mxf_primer_pack_add_mapping (primer, 0x3d07, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->quantization_bits) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &quantization_bits_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->quantization_bits);
+ mxf_primer_pack_add_mapping (primer, 0x3d01, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->dial_norm != 0) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &dial_norm_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->dial_norm);
+ mxf_primer_pack_add_mapping (primer, 0x3d0c, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (!mxf_ul_is_zero (&self->sound_essence_compression)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sound_essence_compression_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->sound_essence_compression, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3d06, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_generic_sound_essence_descriptor_init
(MXFMetadataGenericSoundEssenceDescriptor * self)
{
self->audio_sampling_rate.n = 48000;
self->audio_sampling_rate.d = 1;
+ self->electro_spatial_formulation = 255;
}
static void
@@ -3791,6 +5649,8 @@ static void
MXF_QUARK (GENERIC_SOUND_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_generic_sound_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_generic_sound_essence_descriptor_write_tags;
metadata_class->type = 0x0142;
}
@@ -3817,6 +5677,38 @@ void mxf_metadata_generic_sound_essence_descriptor_set_caps
}
}
+gboolean
+ mxf_metadata_generic_sound_essence_descriptor_from_caps
+ (MXFMetadataGenericSoundEssenceDescriptor * self, GstCaps * caps) {
+ gint rate;
+ gint channels;
+ GstStructure *s;
+
+ g_return_val_if_fail (MXF_IS_METADATA_GENERIC_SOUND_ESSENCE_DESCRIPTOR (self),
+ FALSE);
+ g_return_val_if_fail (GST_IS_CAPS (caps), FALSE);
+
+ s = gst_caps_get_structure (caps, 0);
+
+ if (!gst_structure_get_int (s, "rate", &rate) || rate == 0) {
+ GST_WARNING ("No samplerate");
+ return FALSE;
+ } else {
+ self->audio_sampling_rate.n = rate;
+ self->audio_sampling_rate.d = 1;
+ }
+
+ if (!gst_structure_get_int (s, "channels", &channels) || channels == 0) {
+ GST_WARNING ("No channels");
+ return FALSE;
+ } else {
+ self->channel_count = channels;
+ }
+
+ return TRUE;
+}
+
+
G_DEFINE_TYPE (MXFMetadataCDCIPictureEssenceDescriptor,
mxf_metadata_cdci_picture_essence_descriptor,
MXF_TYPE_METADATA_GENERIC_PICTURE_ESSENCE_DESCRIPTOR);
@@ -3933,7 +5825,7 @@ mxf_metadata_cdci_picture_essence_descriptor_to_structure (MXFMetadataBase * m)
gst_structure_id_set (ret, MXF_QUARK (VERTICAL_SUBSAMPLING), G_TYPE_UINT,
self->vertical_subsampling, NULL);
- if (self->color_siting != 0)
+ if (self->color_siting != 255)
gst_structure_id_set (ret, MXF_QUARK (COLOR_SITING), G_TYPE_UCHAR,
self->color_siting, NULL);
@@ -3963,11 +5855,176 @@ mxf_metadata_cdci_picture_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_cdci_picture_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataCDCIPictureEssenceDescriptor *self =
+ MXF_METADATA_CDCI_PICTURE_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_cdci_picture_essence_descriptor_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 component_depth_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x05, 0x03, 0x0A, 0x00, 0x00, 0x00
+ };
+ static const guint8 horizontal_subsampling_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x05, 0x00, 0x00, 0x00
+ };
+ static const guint8 vertical_subsampling_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x05, 0x01, 0x10, 0x00, 0x00, 0x00
+ };
+ static const guint8 color_siting_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x01, 0x06, 0x00, 0x00, 0x00
+ };
+ static const guint8 reversed_byte_order_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x03, 0x01, 0x02, 0x01, 0x0A, 0x00, 0x00, 0x00
+ };
+ static const guint8 padding_bits_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x18, 0x01, 0x04, 0x00, 0x00, 0x00, 0x00
+ };
+ static const guint8 alpha_sample_depth_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x05, 0x03, 0x07, 0x00, 0x00, 0x00
+ };
+ static const guint8 black_ref_level_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x03, 0x03, 0x00, 0x00, 0x00
+ };
+ static const guint8 white_ref_level_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x04, 0x01, 0x05, 0x03, 0x04, 0x00, 0x00, 0x00
+ };
+ static const guint8 color_range_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x05, 0x03, 0x05, 0x00, 0x00, 0x00
+ };
+
+ if (self->component_depth) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &component_depth_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->component_depth);
+ mxf_primer_pack_add_mapping (primer, 0x3301, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->horizontal_subsampling) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &horizontal_subsampling_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->horizontal_subsampling);
+ mxf_primer_pack_add_mapping (primer, 0x3302, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->vertical_subsampling) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &vertical_subsampling_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->vertical_subsampling);
+ mxf_primer_pack_add_mapping (primer, 0x3308, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->color_siting != 0xff) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &color_siting_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->color_siting);
+ mxf_primer_pack_add_mapping (primer, 0x3303, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->reversed_byte_order) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &reversed_byte_order_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->reversed_byte_order) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0x330b, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->padding_bits) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &padding_bits_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->padding_bits);
+ mxf_primer_pack_add_mapping (primer, 0x3307, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->alpha_sample_depth) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &alpha_sample_depth_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->alpha_sample_depth);
+ mxf_primer_pack_add_mapping (primer, 0x3309, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->black_ref_level) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &black_ref_level_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->black_ref_level);
+ mxf_primer_pack_add_mapping (primer, 0x3304, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->white_ref_level) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &white_ref_level_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->white_ref_level);
+ mxf_primer_pack_add_mapping (primer, 0x3305, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->color_range) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &color_range_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->color_range);
+ mxf_primer_pack_add_mapping (primer, 0x3306, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_cdci_picture_essence_descriptor_init
(MXFMetadataCDCIPictureEssenceDescriptor * self)
{
-
+ self->color_siting = 0xff;
}
static void
@@ -3982,6 +6039,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (CDCI_PICTURE_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_cdci_picture_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_cdci_picture_essence_descriptor_write_tags;
metadata_class->type = 0x0128;
}
@@ -4138,6 +6197,112 @@ mxf_metadata_rgba_picture_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_rgba_picture_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataRGBAPictureEssenceDescriptor *self =
+ MXF_METADATA_RGBA_PICTURE_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_rgba_picture_essence_descriptor_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 component_max_ref_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x05, 0x03, 0x0B, 0x00, 0x00, 0x00
+ };
+ static const guint8 component_min_ref_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x05, 0x03, 0x0C, 0x00, 0x00, 0x00
+ };
+ static const guint8 alpha_max_ref_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x05, 0x03, 0x0D, 0x00, 0x00, 0x00
+ };
+ static const guint8 alpha_min_ref_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x05, 0x03, 0x0E, 0x00, 0x00, 0x00
+ };
+ static const guint8 scanning_direction_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x05,
+ 0x04, 0x01, 0x04, 0x04, 0x01, 0x00, 0x00, 0x00
+ };
+ static const guint8 pixel_layout_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x04, 0x01, 0x05, 0x03, 0x06, 0x00, 0x00, 0x00
+ };
+
+ if (self->component_max_ref != 255) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &component_max_ref_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->component_max_ref);
+ mxf_primer_pack_add_mapping (primer, 0x3406, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->component_min_ref) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &component_min_ref_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->component_min_ref);
+ mxf_primer_pack_add_mapping (primer, 0x3407, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->alpha_max_ref != 255) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &alpha_max_ref_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->alpha_max_ref);
+ mxf_primer_pack_add_mapping (primer, 0x3408, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->alpha_min_ref) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &alpha_min_ref_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->alpha_min_ref);
+ mxf_primer_pack_add_mapping (primer, 0x3409, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->scanning_direction) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &scanning_direction_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->scanning_direction);
+ mxf_primer_pack_add_mapping (primer, 0x3405, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->pixel_layout) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &pixel_layout_ul, 16);
+ t->size = 2 * self->n_pixel_layout + 2;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, self->pixel_layout, self->n_pixel_layout * 2);
+ mxf_primer_pack_add_mapping (primer, 0x3401, &t->key);
+ ret = g_list_prepend (ret, t);
+
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_rgba_picture_essence_descriptor_init
(MXFMetadataRGBAPictureEssenceDescriptor * self)
@@ -4161,6 +6326,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (RGBA_PICTURE_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_rgba_picture_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_rgba_picture_essence_descriptor_write_tags;
metadata_class->type = 0x0129;
}
@@ -4227,6 +6394,36 @@ mxf_metadata_generic_data_essence_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_generic_data_essence_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataGenericDataEssenceDescriptor *self =
+ MXF_METADATA_GENERIC_DATA_ESSENCE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_generic_data_essence_descriptor_parent_class)->write_tags
+ (m, primer);
+ MXFLocalTag *t;
+ static const guint8 data_essence_coding_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x03,
+ 0x04, 0x03, 0x03, 0x02, 0x00, 0x00, 0x00, 0x00
+ };
+
+ if (!mxf_ul_is_zero (&self->data_essence_coding)) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &data_essence_coding_ul, 16);
+ t->size = 16;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ memcpy (t->data, &self->data_essence_coding, 16);
+ mxf_primer_pack_add_mapping (primer, 0x3e01, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_generic_data_essence_descriptor_init
(MXFMetadataGenericDataEssenceDescriptor * self)
@@ -4246,6 +6443,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (GENERIC_DATA_ESSENCE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_generic_data_essence_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_generic_data_essence_descriptor_write_tags;
metadata_class->type = 0x0143;
}
@@ -4391,6 +6590,44 @@ mxf_metadata_multiple_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_multiple_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataMultipleDescriptor *self = MXF_METADATA_MULTIPLE_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_multiple_descriptor_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 sub_descriptors_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x04,
+ 0x06, 0x01, 0x01, 0x04, 0x06, 0x0B, 0x00, 0x00
+ };
+
+ if (self->sub_descriptors) {
+ guint i;
+
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &sub_descriptors_ul, 16);
+ t->size = 8 + 16 * self->n_sub_descriptors;
+ t->data = g_slice_alloc0 (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->n_sub_descriptors);
+ GST_WRITE_UINT32_BE (t->data + 4, 16);
+ for (i = 0; i < self->n_sub_descriptors; i++) {
+ if (!self->sub_descriptors[i])
+ continue;
+
+ memcpy (t->data + 8 + 16 * i,
+ &MXF_METADATA_BASE (self->sub_descriptors[i])->instance_uid, 16);
+ }
+ mxf_primer_pack_add_mapping (primer, 0x3f01, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_multiple_descriptor_init (MXFMetadataMultipleDescriptor * self)
{
@@ -4411,6 +6648,7 @@ mxf_metadata_multiple_descriptor_class_init (MXFMetadataMultipleDescriptorClass
metadata_base_class->name_quark = MXF_QUARK (MULTIPLE_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_multiple_descriptor_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_multiple_descriptor_write_tags;
metadata_class->type = 0x0144;
}
@@ -4480,6 +6718,31 @@ mxf_metadata_text_locator_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_text_locator_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataTextLocator *self = MXF_METADATA_TEXT_LOCATOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_text_locator_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 locator_name_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x02,
+ 0x01, 0x04, 0x01, 0x02, 0x01, 0x00, 0x00, 0x00
+ };
+
+ if (self->locator_name) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &locator_name_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->locator_name, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x4101, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_text_locator_init (MXFMetadataTextLocator * self)
{
@@ -4497,6 +6760,7 @@ mxf_metadata_text_locator_class_init (MXFMetadataTextLocatorClass * klass)
metadata_base_class->handle_tag = mxf_metadata_text_locator_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (TEXT_LOCATOR);
metadata_base_class->to_structure = mxf_metadata_text_locator_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_text_locator_write_tags;
metadata_class->type = 0x0133;
}
@@ -4553,6 +6817,31 @@ mxf_metadata_network_locator_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_network_locator_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataNetworkLocator *self = MXF_METADATA_NETWORK_LOCATOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_network_locator_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+ static const guint8 url_string_ul[] = {
+ 0x06, 0x0E, 0x2B, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x01, 0x02, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00
+ };
+
+ if (self->url_string) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &url_string_ul, 16);
+ t->data = mxf_utf8_to_utf16 (self->url_string, &t->size);
+ mxf_primer_pack_add_mapping (primer, 0x4001, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_network_locator_init (MXFMetadataNetworkLocator * self)
{
@@ -4569,6 +6858,7 @@ mxf_metadata_network_locator_class_init (MXFMetadataNetworkLocatorClass * klass)
metadata_base_class->handle_tag = mxf_metadata_network_locator_handle_tag;
metadata_base_class->name_quark = MXF_QUARK (NETWORK_LOCATOR);
metadata_base_class->to_structure = mxf_metadata_network_locator_to_structure;
+ metadata_base_class->write_tags = mxf_metadata_network_locator_write_tags;
metadata_class->type = 0x0133;
}
diff --git a/gst/mxf/mxfmetadata.h b/gst/mxf/mxfmetadata.h
index 635c6e02..bbdbc62f 100644
--- a/gst/mxf/mxfmetadata.h
+++ b/gst/mxf/mxfmetadata.h
@@ -386,6 +386,7 @@ struct _MXFMetadataBaseClass {
gboolean (*handle_tag) (MXFMetadataBase *self, MXFPrimerPack *primer, guint16 tag, const guint8 *tag_data, guint tag_size);
gboolean (*resolve) (MXFMetadataBase *self, GHashTable *metadata);
GstStructure * (*to_structure) (MXFMetadataBase *self);
+ GList * (*write_tags) (MXFMetadataBase *self, MXFPrimerPack *primer);
GQuark name_quark;
};
@@ -751,15 +752,20 @@ struct _MXFDescriptiveMetadataFrameworkInterface {
gboolean mxf_metadata_base_parse (MXFMetadataBase *self, MXFPrimerPack *primer, const guint8 *data, guint size);
gboolean mxf_metadata_base_resolve (MXFMetadataBase *self, GHashTable *metadata);
GstStructure * mxf_metadata_base_to_structure (MXFMetadataBase *self);
+GstBuffer * mxf_metadata_base_to_buffer (MXFMetadataBase *self, MXFPrimerPack *primer);
MXFMetadata *mxf_metadata_new (guint16 type, MXFPrimerPack *primer, guint64 offset, const guint8 *data, guint size);
void mxf_metadata_register (GType type);
void mxf_metadata_init_types (void);
MXFMetadataTrackType mxf_metadata_track_identifier_parse (const MXFUL * track_identifier);
+const MXFUL * mxf_metadata_track_identifier_get (MXFMetadataTrackType type);
void mxf_metadata_generic_picture_essence_descriptor_set_caps (MXFMetadataGenericPictureEssenceDescriptor * self, GstCaps * caps);
+gboolean mxf_metadata_generic_picture_essence_descriptor_from_caps (MXFMetadataGenericPictureEssenceDescriptor * self, GstCaps * caps);
+
void mxf_metadata_generic_sound_essence_descriptor_set_caps (MXFMetadataGenericSoundEssenceDescriptor * self, GstCaps * caps);
+gboolean mxf_metadata_generic_sound_essence_descriptor_from_caps (MXFMetadataGenericSoundEssenceDescriptor * self, GstCaps * caps);
void mxf_descriptive_metadata_register (guint8 scheme, GType *types);
MXFDescriptiveMetadata * mxf_descriptive_metadata_new (guint8 scheme, guint32 type, MXFPrimerPack * primer, guint64 offset, const guint8 * data, guint size);
diff --git a/gst/mxf/mxfmpeg.c b/gst/mxf/mxfmpeg.c
index 996199af..868aeb80 100644
--- a/gst/mxf/mxfmpeg.c
+++ b/gst/mxf/mxfmpeg.c
@@ -33,10 +33,12 @@
#endif
#include <gst/gst.h>
+#include <gst/video/video.h>
#include <string.h>
#include "mxfmpeg.h"
#include "mxfquark.h"
+#include "mxfwrite.h"
#include <gst/base/gstbytereader.h>
@@ -243,10 +245,133 @@ mxf_metadata_mpeg_video_descriptor_to_structure (MXFMetadataBase * m)
return ret;
}
+static GList *
+mxf_metadata_mpeg_video_descriptor_write_tags (MXFMetadataBase * m,
+ MXFPrimerPack * primer)
+{
+ MXFMetadataMPEGVideoDescriptor *self = MXF_METADATA_MPEG_VIDEO_DESCRIPTOR (m);
+ GList *ret =
+ MXF_METADATA_BASE_CLASS
+ (mxf_metadata_mpeg_video_descriptor_parent_class)->write_tags (m, primer);
+ MXFLocalTag *t;
+
+ if (self->single_sequence != -1) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_single_sequence_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->single_sequence) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->const_b_frames) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_constant_b_frames_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->const_b_frames) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->coded_content_type) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_coded_content_type_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->coded_content_type);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->low_delay) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_low_delay_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->low_delay) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->closed_gop) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_closed_gop_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->closed_gop) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->identical_gop) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_identical_gop_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, (self->identical_gop) ? 1 : 0);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->max_gop) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_identical_gop_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->max_gop);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->b_picture_count) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_b_picture_count_ul, 16);
+ t->size = 2;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT16_BE (t->data, self->b_picture_count);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->bitrate) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_bitrate_ul, 16);
+ t->size = 4;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT32_BE (t->data, self->bitrate);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ if (self->profile_and_level) {
+ t = g_slice_new0 (MXFLocalTag);
+ memcpy (&t->key, &_profile_and_level_ul, 16);
+ t->size = 1;
+ t->data = g_slice_alloc (t->size);
+ t->g_slice = TRUE;
+ GST_WRITE_UINT8 (t->data, self->profile_and_level);
+ mxf_primer_pack_add_mapping (primer, 0, &t->key);
+ ret = g_list_prepend (ret, t);
+ }
+
+ return ret;
+}
+
static void
mxf_metadata_mpeg_video_descriptor_init (MXFMetadataMPEGVideoDescriptor * self)
{
-
+ self->single_sequence = -1;
}
static void
@@ -261,6 +386,8 @@ static void
metadata_base_class->name_quark = MXF_QUARK (MPEG_VIDEO_DESCRIPTOR);
metadata_base_class->to_structure =
mxf_metadata_mpeg_video_descriptor_to_structure;
+ metadata_base_class->write_tags =
+ mxf_metadata_mpeg_video_descriptor_write_tags;
metadata_class->type = 0x0151;
}
@@ -380,8 +507,6 @@ mxf_mpeg_is_mpeg4_keyframe (GstBuffer * buffer)
}
}
- g_assert_not_reached ();
-
return FALSE;
}
@@ -577,7 +702,7 @@ mxf_mpeg_es_create_caps (MXFMetadataTimelineTrack * track, GstTagList ** tags,
codec_name = "MPEG-1 Audio";
} else if (mxf_ul_is_equal (&s->sound_essence_compression,
&sound_essence_compression_ac3)) {
- caps = gst_caps_new_simple ("audio/ac3", NULL);
+ caps = gst_caps_new_simple ("audio/x-ac3", NULL);
codec_name = "AC3 Audio";
} else if (mxf_ul_is_equal (&s->sound_essence_compression,
&sound_essence_compression_mpeg1_layer1)) {
@@ -736,9 +861,431 @@ static const MXFEssenceElementHandler mxf_mpeg_essence_element_handler = {
mxf_mpeg_create_caps
};
+typedef struct
+{
+ guint spf;
+ guint rate;
+} MPEGAudioMappingData;
+
+static GstFlowReturn
+mxf_mpeg_audio_write_func (GstBuffer * buffer, GstCaps * caps,
+ gpointer mapping_data, GstAdapter * adapter, GstBuffer ** outbuf,
+ gboolean flush)
+{
+ *outbuf = buffer;
+ return GST_FLOW_OK;
+}
+
+static const guint8 mpeg_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x02,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x04, 0x01, 0x01
+};
+
+static MXFMetadataFileDescriptor *
+mxf_mpeg_audio_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataGenericSoundEssenceDescriptor *ret;
+ GstStructure *s;
+ MPEGAudioMappingData *md = g_new0 (MPEGAudioMappingData, 1);
+ gint rate;
+
+ md->spf = -1;
+ *mapping_data = md;
+
+ ret = (MXFMetadataGenericSoundEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_GENERIC_SOUND_ESSENCE_DESCRIPTOR);
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "audio/mpeg") == 0) {
+ gint mpegversion;
+
+ if (!gst_structure_get_int (s, "mpegversion", &mpegversion)) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ if (mpegversion == 1) {
+ gint layer = 0;
+ gint mpegaudioversion = 0;
+
+ gst_structure_get_int (s, "layer", &layer);
+ gst_structure_get_int (s, "mpegaudioversion", &mpegaudioversion);
+
+ if (mpegaudioversion == 1 && layer == 1)
+ memcpy (&ret->sound_essence_compression,
+ &sound_essence_compression_mpeg1_layer1, 16);
+ else if (mpegaudioversion == 1 && layer == 2)
+ memcpy (&ret->sound_essence_compression,
+ &sound_essence_compression_mpeg1_layer12, 16);
+ else if (mpegaudioversion == 2 && layer == 1)
+ memcpy (&ret->sound_essence_compression,
+ &sound_essence_compression_mpeg2_layer1, 16);
+
+ if (layer == 1)
+ md->spf = 384;
+ else if (layer == 2 || mpegversion == 1)
+ md->spf = 1152;
+ else
+ md->spf = 576; /* MPEG-2 or 2.5 */
+
+ /* Otherwise all 0x00, must be some kind of mpeg1 audio */
+ } else if (mpegversion == 2) {
+ memcpy (&ret->sound_essence_compression, &sound_essence_compression_aac,
+ 16);
+ md->spf = 1024; /* FIXME: is this correct? */
+ }
+ } else if (strcmp (gst_structure_get_name (s), "audio/x-ac3") == 0) {
+ memcpy (&ret->sound_essence_compression, &sound_essence_compression_ac3,
+ 16);
+ md->spf = 256; /* FIXME: is this correct? */
+ } else {
+ g_assert_not_reached ();
+ }
+
+ if (!gst_structure_get_int (s, "rate", &rate)) {
+ GST_ERROR ("Invalid rate");
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+ md->rate = rate;
+
+ memcpy (&ret->parent.essence_container, &mpeg_essence_container_ul, 16);
+
+ if (!mxf_metadata_generic_sound_essence_descriptor_from_caps (ret, caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_mpeg_audio_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_mpeg_audio_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_mpeg_audio_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ MPEGAudioMappingData *md = mapping_data;
+
+ edit_rate->n = md->rate;
+ edit_rate->d = md->spf;
+}
+
+static guint32
+mxf_mpeg_audio_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x16 << 24) | (0x05 << 8);
+}
+
+static MXFEssenceElementWriter mxf_mpeg_audio_essence_element_writer = {
+ mxf_mpeg_audio_get_descriptor,
+ mxf_mpeg_audio_update_descriptor,
+ mxf_mpeg_audio_get_edit_rate,
+ mxf_mpeg_audio_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
+#define MPEG_AUDIO_CAPS \
+ "audio/mpeg, " \
+ "mpegversion = (int) 1, " \
+ "layer = (int) [ 1, 3 ], " \
+ "rate = (int) [ 8000, 48000 ], " \
+ "channels = (int) [ 1, 2 ], " \
+ "parsed = (boolean) TRUE; " \
+ "audio/x-ac3, " \
+ "rate = (int) [ 4000, 96000 ], " \
+ "channels = (int) [ 1, 6 ]; " \
+ "audio/mpeg, " \
+ "mpegversion = (int) 2, " \
+ "rate = (int) [ 8000, 96000 ], " \
+ "channels = (int) [ 1, 8 ]"
+
+/* See ISO/IEC 13818-2 for MPEG ES format */
+gboolean
+mxf_mpeg_is_mpeg2_frame (GstBuffer * buffer)
+{
+ GstByteReader reader = GST_BYTE_READER_INIT_FROM_BUFFER (buffer);
+ guint32 tmp;
+
+ while (gst_byte_reader_get_remaining (&reader) > 3) {
+ if (gst_byte_reader_peek_uint24_be (&reader, &tmp) && tmp == 0x000001) {
+ guint8 type;
+
+ /* Found sync code */
+ gst_byte_reader_skip (&reader, 3);
+
+ if (!gst_byte_reader_get_uint8 (&reader, &type))
+ break;
+
+ /* PICTURE */
+ if (type == 0x00) {
+ return TRUE;
+ }
+ } else {
+ gst_byte_reader_skip (&reader, 1);
+ }
+ }
+
+ return FALSE;
+}
+
+static gboolean
+mxf_mpeg_is_mpeg4_frame (GstBuffer * buffer)
+{
+ GstByteReader reader = GST_BYTE_READER_INIT_FROM_BUFFER (buffer);
+ guint32 tmp;
+
+ while (gst_byte_reader_get_remaining (&reader) > 3) {
+ if (gst_byte_reader_peek_uint24_be (&reader, &tmp) && tmp == 0x000001) {
+ guint8 type;
+
+ /* Found sync code */
+ gst_byte_reader_skip (&reader, 3);
+
+ if (!gst_byte_reader_get_uint8 (&reader, &type))
+ break;
+
+ /* PICTURE */
+ if (type == 0xb6) {
+ return TRUE;
+ }
+ } else {
+ gst_byte_reader_skip (&reader, 1);
+ }
+ }
+
+ return FALSE;
+}
+
+static GstFlowReturn
+mxf_mpeg_video_write_func (GstBuffer * buffer, GstCaps * caps,
+ gpointer mapping_data, GstAdapter * adapter, GstBuffer ** outbuf,
+ gboolean flush)
+{
+ MXFMPEGEssenceType type = MXF_MPEG_ESSENCE_TYPE_OTHER;
+
+ if (mapping_data)
+ type = *((MXFMPEGEssenceType *) mapping_data);
+
+ if (type == MXF_MPEG_ESSENCE_TYPE_VIDEO_MPEG2) {
+ if (buffer && !mxf_mpeg_is_mpeg2_frame (buffer)) {
+ gst_adapter_push (adapter, buffer);
+ *outbuf = NULL;
+ return GST_FLOW_OK;
+ } else if (buffer || gst_adapter_available (adapter)) {
+ guint av = gst_adapter_available (adapter);
+ GstBuffer *ret;
+
+ if (buffer)
+ ret = gst_buffer_new_and_alloc (GST_BUFFER_SIZE (buffer) + av);
+ else
+ ret = gst_buffer_new_and_alloc (av);
+
+ if (av) {
+ GstBuffer *tmp = gst_adapter_take_buffer (adapter, av);
+ memcpy (GST_BUFFER_DATA (ret), GST_BUFFER_DATA (tmp), av);
+ gst_buffer_unref (tmp);
+ }
+
+ if (buffer) {
+ memcpy (GST_BUFFER_DATA (ret) + av, GST_BUFFER_DATA (buffer),
+ GST_BUFFER_SIZE (buffer));
+ gst_buffer_unref (buffer);
+ }
+ *outbuf = ret;
+ return GST_FLOW_OK;
+ }
+ } else if (type == MXF_MPEG_ESSENCE_TYPE_VIDEO_MPEG4) {
+ if (buffer && !mxf_mpeg_is_mpeg4_frame (buffer)) {
+ gst_adapter_push (adapter, buffer);
+ *outbuf = NULL;
+ return GST_FLOW_OK;
+ } else if (buffer || gst_adapter_available (adapter)) {
+ guint av = gst_adapter_available (adapter);
+ GstBuffer *ret;
+
+ if (buffer)
+ ret = gst_buffer_new_and_alloc (GST_BUFFER_SIZE (buffer) + av);
+ else
+ ret = gst_buffer_new_and_alloc (av);
+
+ if (av) {
+ GstBuffer *tmp = gst_adapter_take_buffer (adapter, av);
+ memcpy (GST_BUFFER_DATA (ret), GST_BUFFER_DATA (tmp), av);
+ gst_buffer_unref (tmp);
+ }
+
+ if (buffer) {
+ memcpy (GST_BUFFER_DATA (ret) + av, GST_BUFFER_DATA (buffer),
+ GST_BUFFER_SIZE (buffer));
+ gst_buffer_unref (buffer);
+ }
+ *outbuf = ret;
+ return GST_FLOW_OK;
+ }
+ }
+
+ *outbuf = buffer;
+ return GST_FLOW_OK;
+}
+
+static const guint8 mpeg_video_picture_essence_compression_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x00,
+ 0x04, 0x01, 0x02, 0x02, 0x01, 0x00, 0x00, 0x00
+};
+
+static MXFMetadataFileDescriptor *
+mxf_mpeg_video_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataMPEGVideoDescriptor *ret;
+ GstStructure *s;
+
+ ret = (MXFMetadataMPEGVideoDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_MPEG_VIDEO_DESCRIPTOR);
+
+ s = gst_caps_get_structure (caps, 0);
+
+ memcpy (&ret->parent.parent.parent.essence_container,
+ &mpeg_essence_container_ul, 16);
+ memcpy (&ret->parent.parent.picture_essence_coding,
+ &mpeg_video_picture_essence_compression_ul, 16);
+ if (strcmp (gst_structure_get_name (s), "video/mpeg") == 0) {
+ gint mpegversion;
+
+ if (!gst_structure_get_int (s, "mpegversion", &mpegversion)) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ if (mpegversion == 1) {
+ MXFMPEGEssenceType type = MXF_MPEG_ESSENCE_TYPE_VIDEO_MPEG2;
+
+ *mapping_data = g_new0 (MXFMPEGEssenceType, 1);
+ memcpy (*mapping_data, &type, sizeof (MXFMPEGEssenceType));
+ ret->parent.parent.picture_essence_coding.u[13] = 0x10;
+ } else if (mpegversion == 2) {
+ MXFMPEGEssenceType type = MXF_MPEG_ESSENCE_TYPE_VIDEO_MPEG2;
+
+ *mapping_data = g_new0 (MXFMPEGEssenceType, 1);
+ memcpy (*mapping_data, &type, sizeof (MXFMPEGEssenceType));
+ ret->parent.parent.picture_essence_coding.u[13] = 0x01;
+ } else {
+ const GValue *v;
+ const GstBuffer *codec_data;
+ MXFMPEGEssenceType type = MXF_MPEG_ESSENCE_TYPE_VIDEO_MPEG4;
+
+ *mapping_data = g_new0 (MXFMPEGEssenceType, 1);
+ memcpy (*mapping_data, &type, sizeof (MXFMPEGEssenceType));
+
+ ret->parent.parent.picture_essence_coding.u[13] = 0x20;
+ if ((v = gst_structure_get_value (s, "codec_data"))) {
+ MXFLocalTag *t = g_slice_new0 (MXFLocalTag);
+ codec_data = gst_value_get_buffer (v);
+ t->size = GST_BUFFER_SIZE (codec_data);
+ t->data = g_memdup (GST_BUFFER_DATA (codec_data), t->size);
+ memcpy (&t->key, &sony_mpeg4_extradata, 16);
+ mxf_local_tag_insert (t, &MXF_METADATA_BASE (ret)->other_tags);
+ }
+ }
+ } else if (strcmp (gst_structure_get_name (s), "video/x-h264") == 0) {
+ MXFMPEGEssenceType type = MXF_MPEG_ESSENCE_TYPE_VIDEO_AVC;
+
+ *mapping_data = g_new0 (MXFMPEGEssenceType, 1);
+ memcpy (*mapping_data, &type, sizeof (MXFMPEGEssenceType));
+ ret->parent.parent.picture_essence_coding.u[13] = 0x30;
+ } else {
+ g_assert_not_reached ();
+ }
+
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->
+ parent.parent, caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_mpeg_video_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_mpeg_video_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_mpeg_video_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ (*edit_rate).n = a->sample_rate.n;
+ (*edit_rate).d = a->sample_rate.d;
+}
+
+static guint32
+mxf_mpeg_video_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x15 << 24) | (0x05 << 8);
+}
+
+static MXFEssenceElementWriter mxf_mpeg_video_essence_element_writer = {
+ mxf_mpeg_video_get_descriptor,
+ mxf_mpeg_video_update_descriptor,
+ mxf_mpeg_video_get_edit_rate,
+ mxf_mpeg_video_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
+#define MPEG_VIDEO_CAPS \
+"video/mpeg, " \
+"mpegversion = (int) { 1, 2, 4 }, " \
+"systemstream = (boolean) FALSE, " \
+"width = " GST_VIDEO_SIZE_RANGE ", " \
+"height = " GST_VIDEO_SIZE_RANGE ", " \
+"framerate = " GST_VIDEO_FPS_RANGE "; " \
+"video/x-h264, " \
+"width = " GST_VIDEO_SIZE_RANGE ", " \
+"height = " GST_VIDEO_SIZE_RANGE ", " \
+"framerate = " GST_VIDEO_FPS_RANGE
+
void
mxf_mpeg_init (void)
{
mxf_metadata_register (MXF_TYPE_METADATA_MPEG_VIDEO_DESCRIPTOR);
mxf_essence_element_handler_register (&mxf_mpeg_essence_element_handler);
+
+ mxf_mpeg_audio_essence_element_writer.pad_template =
+ gst_pad_template_new ("mpeg_audio_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string (MPEG_AUDIO_CAPS));
+ memcpy (&mxf_mpeg_audio_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_SOUND_ESSENCE), 16);
+ mxf_essence_element_writer_register (&mxf_mpeg_audio_essence_element_writer);
+
+ mxf_mpeg_video_essence_element_writer.pad_template =
+ gst_pad_template_new ("mpeg_video_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string (MPEG_VIDEO_CAPS));
+ memcpy (&mxf_mpeg_video_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_PICTURE_ESSENCE),
+ 16);
+ mxf_essence_element_writer_register (&mxf_mpeg_video_essence_element_writer);
+
}
diff --git a/gst/mxf/mxfmux.c b/gst/mxf/mxfmux.c
new file mode 100644
index 00000000..0052d2e1
--- /dev/null
+++ b/gst/mxf/mxfmux.c
@@ -0,0 +1,1429 @@
+/* GStreamer
+ * Copyright (C) 2009 Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <math.h>
+#include <string.h>
+
+#include "mxfmux.h"
+
+#ifdef HAVE_SYS_UTSNAME_H
+#include <sys/utsname.h>
+#endif
+
+GST_DEBUG_CATEGORY_STATIC (mxfmux_debug);
+#define GST_CAT_DEFAULT mxfmux_debug
+
+static GstStaticPadTemplate src_templ = GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("application/mxf")
+ );
+
+enum
+{
+ PROP_0
+};
+
+GST_BOILERPLATE (GstMXFMux, gst_mxf_mux, GstElement, GST_TYPE_ELEMENT);
+
+static void gst_mxf_mux_finalize (GObject * object);
+static void gst_mxf_mux_set_property (GObject * object,
+ guint prop_id, const GValue * value, GParamSpec * pspec);
+static void gst_mxf_mux_get_property (GObject * object,
+ guint prop_id, GValue * value, GParamSpec * pspec);
+
+static GstFlowReturn gst_mxf_mux_collected (GstCollectPads * pads,
+ gpointer user_data);
+
+static gboolean gst_mxf_mux_handle_src_event (GstPad * pad, GstEvent * event);
+static GstPad *gst_mxf_mux_request_new_pad (GstElement * element,
+ GstPadTemplate * templ, const gchar * name);
+static void gst_mxf_mux_release_pad (GstElement * element, GstPad * pad);
+
+static GstStateChangeReturn
+gst_mxf_mux_change_state (GstElement * element, GstStateChange transition);
+
+static void gst_mxf_mux_reset (GstMXFMux * mux);
+
+static GstFlowReturn
+gst_mxf_mux_push (GstMXFMux * mux, GstBuffer * buf)
+{
+ guint size = GST_BUFFER_SIZE (buf);
+ GstFlowReturn ret;
+
+ gst_buffer_set_caps (buf, GST_PAD_CAPS (mux->srcpad));
+ ret = gst_pad_push (mux->srcpad, buf);
+ mux->offset += size;
+
+ return ret;
+}
+
+static void
+gst_mxf_mux_base_init (gpointer g_class)
+{
+ GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+ const GstPadTemplate **p;
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&src_templ));
+
+ p = mxf_essence_element_writer_get_pad_templates ();
+ while (p && *p) {
+ gst_element_class_add_pad_template (element_class,
+ (GstPadTemplate *) gst_object_ref (GST_OBJECT (*p)));
+ p++;
+ }
+
+ gst_element_class_set_details_simple (element_class, "MXF muxer",
+ "Codec/Muxer",
+ "Muxes video/audio streams into a MXF stream",
+ "Sebastian Dröge <sebastian.droege@collabora.co.uk>");
+}
+
+static void
+gst_mxf_mux_class_init (GstMXFMuxClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *gstelement_class;
+
+ GST_DEBUG_CATEGORY_INIT (mxfmux_debug, "mxfmux", 0, "MXF muxer");
+
+ gobject_class = (GObjectClass *) klass;
+ gstelement_class = (GstElementClass *) klass;
+
+ gobject_class->finalize = gst_mxf_mux_finalize;
+ gobject_class->set_property = gst_mxf_mux_set_property;
+ gobject_class->get_property = gst_mxf_mux_get_property;
+
+ gstelement_class->change_state = GST_DEBUG_FUNCPTR (gst_mxf_mux_change_state);
+ gstelement_class->request_new_pad =
+ GST_DEBUG_FUNCPTR (gst_mxf_mux_request_new_pad);
+ gstelement_class->release_pad = GST_DEBUG_FUNCPTR (gst_mxf_mux_release_pad);
+}
+
+static void
+gst_mxf_mux_init (GstMXFMux * mux, GstMXFMuxClass * g_class)
+{
+ GstCaps *caps;
+
+ mux->srcpad = gst_pad_new_from_static_template (&src_templ, "src");
+ gst_pad_set_event_function (mux->srcpad, gst_mxf_mux_handle_src_event);
+ caps = gst_caps_new_simple ("application/mxf", NULL);
+ gst_pad_set_caps (mux->srcpad, caps);
+ gst_caps_unref (caps);
+ gst_element_add_pad (GST_ELEMENT (mux), mux->srcpad);
+
+ mux->collect = gst_collect_pads_new ();
+ gst_collect_pads_set_function (mux->collect,
+ (GstCollectPadsFunction) GST_DEBUG_FUNCPTR (gst_mxf_mux_collected), mux);
+
+ gst_mxf_mux_reset (mux);
+}
+
+static void
+gst_mxf_mux_finalize (GObject * object)
+{
+ GstMXFMux *mux = GST_MXF_MUX (object);
+
+ gst_mxf_mux_reset (mux);
+
+ if (mux->metadata) {
+ g_hash_table_destroy (mux->metadata);
+ mux->metadata = NULL;
+ }
+
+ gst_object_unref (mux->collect);
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+gst_mxf_mux_set_property (GObject * object,
+ guint prop_id, const GValue * value, GParamSpec * pspec)
+{
+ //GstMXFMux *mux = GST_MXF_MUX (object);
+
+ switch (prop_id) {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_mxf_mux_get_property (GObject * object,
+ guint prop_id, GValue * value, GParamSpec * pspec)
+{
+ //GstMXFMux *mux = GST_MXF_MUX (object);
+
+ switch (prop_id) {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_mxf_mux_reset (GstMXFMux * mux)
+{
+ GSList *sl;
+
+ while ((sl = mux->collect->data) != NULL) {
+ GstMXFMuxPad *cpad = (GstMXFMuxPad *) sl->data;
+
+ gst_object_unref (cpad->adapter);
+ g_free (cpad->mapping_data);
+
+ gst_collect_pads_remove_pad (mux->collect, cpad->collect.pad);
+ }
+
+ mux->state = GST_MXF_MUX_STATE_HEADER;
+ mux->n_pads = 0;
+
+ if (mux->metadata) {
+ g_hash_table_destroy (mux->metadata);
+ mux->preface = NULL;
+ }
+ mux->metadata = mxf_metadata_hash_table_new ();
+
+ mxf_partition_pack_reset (&mux->partition);
+ mxf_primer_pack_reset (&mux->primer);
+ memset (&mux->min_edit_rate, 0, sizeof (MXFFraction));
+ mux->last_gc_timestamp = 0;
+ mux->last_gc_position = 0;
+ mux->offset = 0;
+}
+
+static gboolean
+gst_mxf_mux_handle_src_event (GstPad * pad, GstEvent * event)
+{
+ GstEventType type;
+
+ type = event ? GST_EVENT_TYPE (event) : GST_EVENT_UNKNOWN;
+
+ switch (type) {
+ case GST_EVENT_SEEK:
+ /* disable seeking for now */
+ return FALSE;
+ default:
+ break;
+ }
+
+ return gst_pad_event_default (pad, event);
+}
+
+static gboolean
+gst_mxf_mux_handle_sink_event (GstPad * pad, GstEvent * event)
+{
+ GstMXFMux *mux = GST_MXF_MUX (gst_pad_get_parent (pad));
+ gboolean ret = TRUE;
+
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_TAG:
+ /* TODO: do something with the tags */
+ break;
+ case GST_EVENT_NEWSEGMENT:
+ /* We don't support NEWSEGMENT events */
+ ret = FALSE;
+ gst_event_unref (event);
+ break;
+ default:
+ break;
+ }
+
+ /* now GstCollectPads can take care of the rest, e.g. EOS */
+ if (ret)
+ ret = mux->collect_event (pad, event);
+ gst_object_unref (mux);
+
+ return ret;
+}
+
+static gboolean
+gst_mxf_mux_setcaps (GstPad * pad, GstCaps * caps)
+{
+ GstMXFMux *mux = GST_MXF_MUX (gst_pad_get_parent (pad));
+ GstMXFMuxPad *cpad = (GstMXFMuxPad *) gst_pad_get_element_private (pad);
+ gboolean ret = TRUE;
+ MXFUL d_instance_uid = { {0,} };
+ MXFMetadataFileDescriptor *old_descriptor = cpad->descriptor;
+
+ GST_DEBUG_OBJECT (pad, "Setting caps %" GST_PTR_FORMAT, caps);
+
+ if (old_descriptor) {
+ memcpy (&d_instance_uid, &MXF_METADATA_BASE (old_descriptor)->instance_uid,
+ 16);
+ cpad->descriptor = NULL;
+ g_free (cpad->mapping_data);
+ cpad->mapping_data = NULL;
+ }
+
+ cpad->descriptor =
+ cpad->writer->get_descriptor (gst_pad_get_pad_template (pad), caps,
+ &cpad->write_func, &cpad->mapping_data);
+
+ if (!cpad->descriptor) {
+ GST_ERROR_OBJECT (mux,
+ "Couldn't get descriptor for pad '%s' with caps %" GST_PTR_FORMAT,
+ GST_PAD_NAME (pad), caps);
+ gst_object_unref (mux);
+ return FALSE;
+ }
+
+ if (mxf_ul_is_zero (&d_instance_uid))
+ mxf_ul_set (&d_instance_uid, mux->metadata);
+
+ memcpy (&MXF_METADATA_BASE (cpad->descriptor)->instance_uid, &d_instance_uid,
+ 16);
+
+ g_hash_table_replace (mux->metadata,
+ &MXF_METADATA_BASE (cpad->descriptor)->instance_uid, cpad->descriptor);
+
+ if (old_descriptor) {
+ if (mux->preface && mux->preface->content_storage &&
+ mux->preface->content_storage->packages) {
+ guint i, j;
+
+ for (i = 0; i < mux->preface->content_storage->n_packages; i++) {
+ MXFMetadataSourcePackage *package;
+
+ if (!MXF_IS_METADATA_SOURCE_PACKAGE (mux->preface->content_storage->
+ packages[i]))
+ continue;
+
+ package =
+ MXF_METADATA_SOURCE_PACKAGE (mux->preface->content_storage->
+ packages[i]);
+
+ if (!package->descriptor)
+ continue;
+
+ if (MXF_IS_METADATA_MULTIPLE_DESCRIPTOR (package->descriptor)) {
+ MXFMetadataMultipleDescriptor *tmp =
+ MXF_METADATA_MULTIPLE_DESCRIPTOR (package->descriptor);
+
+ for (j = 0; j < tmp->n_sub_descriptors; j++) {
+ if (tmp->sub_descriptors[j] ==
+ MXF_METADATA_GENERIC_DESCRIPTOR (old_descriptor)) {
+ tmp->sub_descriptors[j] =
+ MXF_METADATA_GENERIC_DESCRIPTOR (cpad->descriptor);
+ memcpy (&tmp->sub_descriptors_uids[j], &d_instance_uid, 16);
+ }
+ }
+ } else if (package->descriptor ==
+ MXF_METADATA_GENERIC_DESCRIPTOR (old_descriptor)) {
+ package->descriptor =
+ MXF_METADATA_GENERIC_DESCRIPTOR (cpad->descriptor);
+ memcpy (&package->descriptor_uid, &d_instance_uid, 16);
+ }
+ }
+ }
+ }
+
+ gst_object_unref (mux);
+
+ return ret;
+}
+
+static GstPad *
+gst_mxf_mux_request_new_pad (GstElement * element,
+ GstPadTemplate * templ, const gchar * pad_name)
+{
+ GstMXFMux *mux = GST_MXF_MUX (element);
+ GstMXFMuxPad *cpad;
+ GstPad *pad = NULL;
+ guint pad_number;
+ gchar *name = NULL;
+ const MXFEssenceElementWriter *writer;
+
+ if (mux->state != GST_MXF_MUX_STATE_HEADER) {
+ GST_WARNING_OBJECT (mux, "Can't request pads after writing header");
+ return NULL;
+ }
+
+ writer = mxf_essence_element_writer_find (templ);
+ if (!writer) {
+ GST_ERROR_OBJECT (mux, "Not our template");
+ return NULL;
+ }
+
+ pad_number = g_atomic_int_exchange_and_add ((gint *) & mux->n_pads, 1);
+ name = g_strdup_printf (GST_PAD_TEMPLATE_NAME_TEMPLATE (templ), pad_number);
+
+ GST_DEBUG_OBJECT (mux, "Creating pad '%s'", name);
+ pad = gst_pad_new_from_template (templ, name);
+ g_free (name);
+ cpad = (GstMXFMuxPad *)
+ gst_collect_pads_add_pad (mux->collect, pad, sizeof (GstMXFMuxPad));
+ cpad->last_timestamp = 0;
+ cpad->adapter = gst_adapter_new ();
+ cpad->writer = writer;
+
+ /* FIXME: hacked way to override/extend the event function of
+ * GstCollectPads; because it sets its own event function giving the
+ * element no access to events.
+ */
+ mux->collect_event = (GstPadEventFunction) GST_PAD_EVENTFUNC (pad);
+ gst_pad_set_event_function (pad,
+ GST_DEBUG_FUNCPTR (gst_mxf_mux_handle_sink_event));
+
+ gst_pad_set_setcaps_function (pad, gst_mxf_mux_setcaps);
+ gst_pad_use_fixed_caps (pad);
+ gst_pad_set_active (pad, TRUE);
+ gst_element_add_pad (element, pad);
+
+ return pad;
+}
+
+static void
+gst_mxf_mux_release_pad (GstElement * element, GstPad * pad)
+{
+ /*GstMXFMux *mux = GST_MXF_MUX (GST_PAD_PARENT (pad));
+ GstMXFMuxPad *cpad = (GstMXFMuxPad *) gst_pad_get_element_private (pad);
+
+ gst_object_unref (cpad->adapter);
+ g_free (cpad->mapping_data);
+
+ gst_collect_pads_remove_pad (mux->collect, pad);
+ gst_element_remove_pad (element, pad); */
+}
+
+static GstFlowReturn
+gst_mxf_mux_create_metadata (GstMXFMux * mux)
+{
+ GstFlowReturn ret = GST_FLOW_OK;
+ GSList *l;
+ GArray *tmp;
+
+ GST_DEBUG_OBJECT (mux, "Creating MXF metadata");
+
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+
+ if (!cpad || !cpad->descriptor || !GST_PAD_CAPS (cpad->collect.pad))
+ return GST_FLOW_ERROR;
+
+ if (cpad->writer->update_descriptor)
+ cpad->writer->update_descriptor (cpad->descriptor,
+ GST_PAD_CAPS (cpad->collect.pad), cpad->mapping_data,
+ cpad->collect.buffer);
+ }
+
+ /* Preface */
+ mux->preface =
+ (MXFMetadataPreface *) gst_mini_object_new (MXF_TYPE_METADATA_PREFACE);
+ mxf_ul_set (&MXF_METADATA_BASE (mux->preface)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (mux->preface)->instance_uid, mux->preface);
+
+ mxf_timestamp_set_now (&mux->preface->last_modified_date);
+ mux->preface->version = 258;
+ mux->preface->object_model_version = 1;
+
+ mxf_op_set_generalized (&mux->preface->operational_pattern, MXF_OP_1a, TRUE,
+ TRUE, FALSE);
+
+ tmp = g_array_new (FALSE, FALSE, sizeof (MXFUL));
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ guint i;
+ gboolean found = FALSE;
+
+ if (!cpad || !cpad->descriptor ||
+ mxf_ul_is_zero (&cpad->descriptor->essence_container))
+ return GST_FLOW_ERROR;
+
+ for (i = 0; i < tmp->len; i++) {
+ if (mxf_ul_is_equal (&cpad->descriptor->essence_container,
+ &g_array_index (tmp, MXFUL, i))) {
+ found = TRUE;
+ break;
+ }
+ }
+
+ if (found)
+ continue;
+
+ g_array_append_val (tmp, cpad->descriptor->essence_container);
+ }
+ mux->preface->n_essence_containers = tmp->len;
+ mux->preface->essence_containers = (MXFUL *) g_array_free (tmp, FALSE);
+
+ /* This will later be used as UID for the material package */
+ mxf_ul_set (&mux->preface->primary_package_uid, mux->metadata);
+
+ /* Identifications */
+ {
+ MXFMetadataIdentification *identification;
+ static const guint8 gst_uid[] = {
+ 0xe5, 0xde, 0xcd, 0x04, 0x24, 0x90, 0x69, 0x18,
+ 0x8a, 0xc9, 0xb5, 0xd7, 0x02, 0x58, 0x46, 0x78
+ };
+ guint major, minor, micro, nano;
+
+ mux->preface->n_identifications = 1;
+ mux->preface->identifications = g_new0 (MXFMetadataIdentification *, 1);
+ identification = mux->preface->identifications[0] =
+ (MXFMetadataIdentification *)
+ gst_mini_object_new (MXF_TYPE_METADATA_IDENTIFICATION);
+
+ mxf_ul_set (&MXF_METADATA_BASE (identification)->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (identification)->instance_uid, identification);
+
+ mxf_ul_set (&identification->this_generation_uid, NULL);
+
+ identification->company_name = g_strdup ("GStreamer");
+ identification->product_name = g_strdup ("GStreamer Multimedia Framework");
+
+ gst_version (&major, &minor, &micro, &nano);
+ identification->product_version.major = major;
+ identification->product_version.minor = minor;
+ identification->product_version.patch = micro;
+ identification->product_version.build = nano;
+ identification->product_version.release =
+ (nano == 0) ? 1 : (nano == 1) ? 2 : 4;
+
+ identification->version_string =
+ g_strdup_printf ("%u.%u.%u.%u", major, minor, micro, nano);
+ memcpy (&identification->product_uid, &gst_uid, 16);
+
+ memcpy (&identification->modification_date,
+ &mux->preface->last_modified_date, sizeof (MXFTimestamp));
+ memcpy (&identification->toolkit_version, &identification->product_version,
+ sizeof (MXFProductVersion));
+
+#ifdef HAVE_SYS_UTSNAME_H
+ {
+ struct utsname sys_details;
+
+ if (uname (&sys_details) == 0) {
+ identification->platform = g_strdup_printf ("%s %s %s",
+ sys_details.sysname, sys_details.release, sys_details.machine);
+ }
+ }
+#endif
+
+#if defined(G_OS_WIN32)
+ if (identification->platform == NULL)
+ identification->platform = g_strdup ("Microsoft Windows");
+#elif defined(G_OS_BEOS)
+ if (identification->platform == NULL)
+ identification->platform = g_strdup ("BEOS");
+#elif defined(G_OS_UNIX)
+ if (identification->platform == NULL)
+ identification->platform = g_strdup ("Unix");
+#endif
+ }
+
+ /* Content storage */
+ {
+ MXFMetadataContentStorage *cstorage;
+ guint i;
+
+ cstorage = mux->preface->content_storage = (MXFMetadataContentStorage *)
+ gst_mini_object_new (MXF_TYPE_METADATA_CONTENT_STORAGE);
+ mxf_ul_set (&MXF_METADATA_BASE (cstorage)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (cstorage)->instance_uid, cstorage);
+
+ cstorage->n_packages = 2;
+ cstorage->packages = g_new0 (MXFMetadataGenericPackage *, 2);
+
+ /* Source package */
+ {
+ MXFMetadataSourcePackage *p;
+
+ cstorage->packages[1] = (MXFMetadataGenericPackage *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SOURCE_PACKAGE);
+ mxf_ul_set (&MXF_METADATA_BASE (cstorage->packages[1])->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (cstorage->packages[1])->instance_uid,
+ cstorage->packages[1]);
+ p = (MXFMetadataSourcePackage *) cstorage->packages[1];
+
+ mxf_umid_set (&p->parent.package_uid);
+ p->parent.name = g_strdup ("Source package");
+ memcpy (&p->parent.package_creation_date,
+ &mux->preface->last_modified_date, sizeof (MXFTimestamp));
+ memcpy (&p->parent.package_modified_date,
+ &mux->preface->last_modified_date, sizeof (MXFTimestamp));
+
+ p->parent.n_tracks = g_slist_length (mux->collect->data);
+ p->parent.tracks = g_new0 (MXFMetadataTrack *, p->parent.n_tracks);
+
+ if (p->parent.n_tracks > 1) {
+ MXFMetadataMultipleDescriptor *d;
+
+ p->descriptor = (MXFMetadataGenericDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_MULTIPLE_DESCRIPTOR);
+ d = (MXFMetadataMultipleDescriptor *) p->descriptor;
+ d->n_sub_descriptors = p->parent.n_tracks;
+ d->sub_descriptors =
+ g_new0 (MXFMetadataGenericDescriptor *, p->parent.n_tracks);
+
+ mxf_ul_set (&MXF_METADATA_BASE (d)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (d)->instance_uid, d);
+ }
+
+ /* Tracks */
+ {
+ guint n = 0;
+
+ /* Essence tracks */
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ MXFMetadataTimelineTrack *track;
+ MXFMetadataSequence *sequence;
+ MXFMetadataSourceClip *clip;
+
+ p->parent.tracks[n] = (MXFMetadataTrack *)
+ gst_mini_object_new (MXF_TYPE_METADATA_TIMELINE_TRACK);
+ track = (MXFMetadataTimelineTrack *) p->parent.tracks[n];
+ mxf_ul_set (&MXF_METADATA_BASE (track)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (track)->instance_uid, track);
+
+ track->parent.track_id = n + 1;
+ track->parent.track_number =
+ cpad->writer->get_track_number_template (cpad->descriptor,
+ GST_PAD_CAPS (cpad->collect.pad), cpad->mapping_data);
+
+ cpad->writer->get_edit_rate (cpad->descriptor,
+ GST_PAD_CAPS (cpad->collect.pad), cpad->mapping_data,
+ cpad->collect.buffer, p, track, &track->edit_rate);
+
+ sequence = track->parent.sequence = (MXFMetadataSequence *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SEQUENCE);
+ mxf_ul_set (&MXF_METADATA_BASE (sequence)->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (sequence)->instance_uid, sequence);
+
+ memcpy (&sequence->data_definition, &cpad->writer->data_definition,
+ 16);
+
+ sequence->n_structural_components = 1;
+ sequence->structural_components =
+ g_new0 (MXFMetadataStructuralComponent *, 1);
+
+ clip = (MXFMetadataSourceClip *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SOURCE_CLIP);
+ sequence->structural_components[0] =
+ (MXFMetadataStructuralComponent *) clip;
+ mxf_ul_set (&MXF_METADATA_BASE (clip)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (clip)->instance_uid, clip);
+
+ memcpy (&clip->parent.data_definition, &sequence->data_definition,
+ 16);
+ clip->start_position = 0;
+
+ cpad->source_package = p;
+ cpad->source_track = track;
+ cpad->descriptor->linked_track_id = n + 1;
+ if (p->parent.n_tracks == 1) {
+ p->descriptor = (MXFMetadataGenericDescriptor *) cpad->descriptor;
+ } else {
+ MXF_METADATA_MULTIPLE_DESCRIPTOR (p->descriptor)->
+ sub_descriptors[n] =
+ (MXFMetadataGenericDescriptor *) cpad->descriptor;
+ }
+
+ n++;
+ }
+ }
+ }
+
+ /* Material package */
+ {
+ MXFMetadataMaterialPackage *p;
+ MXFFraction min_edit_rate = { 0, 0 };
+ gdouble min_edit_rate_d = G_MAXDOUBLE;
+
+ cstorage->packages[0] = (MXFMetadataGenericPackage *)
+ gst_mini_object_new (MXF_TYPE_METADATA_MATERIAL_PACKAGE);
+ memcpy (&MXF_METADATA_BASE (cstorage->packages[0])->instance_uid,
+ &mux->preface->primary_package_uid, 16);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (cstorage->packages[0])->instance_uid,
+ cstorage->packages[0]);
+ p = (MXFMetadataMaterialPackage *) cstorage->packages[0];
+
+ mxf_umid_set (&p->package_uid);
+ p->name = g_strdup ("Material package");
+ memcpy (&p->package_creation_date, &mux->preface->last_modified_date,
+ sizeof (MXFTimestamp));
+ memcpy (&p->package_modified_date, &mux->preface->last_modified_date,
+ sizeof (MXFTimestamp));
+
+ p->n_tracks = g_slist_length (mux->collect->data) + 1;
+ p->tracks = g_new0 (MXFMetadataTrack *, p->n_tracks);
+
+ /* Tracks */
+ {
+ guint n;
+
+ n = 1;
+ /* Essence tracks */
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ MXFMetadataSourcePackage *source_package;
+ MXFMetadataTimelineTrack *track, *source_track;
+ MXFMetadataSequence *sequence;
+ MXFMetadataSourceClip *clip;
+
+ source_package = MXF_METADATA_SOURCE_PACKAGE (cstorage->packages[1]);
+ source_track =
+ MXF_METADATA_TIMELINE_TRACK (source_package->parent.tracks[n -
+ 1]);
+
+ p->tracks[n] = (MXFMetadataTrack *)
+ gst_mini_object_new (MXF_TYPE_METADATA_TIMELINE_TRACK);
+ track = (MXFMetadataTimelineTrack *) p->tracks[n];
+ mxf_ul_set (&MXF_METADATA_BASE (track)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (track)->instance_uid, track);
+
+ track->parent.track_id = n + 1;
+ track->parent.track_number = 0;
+
+ cpad->writer->get_edit_rate (cpad->descriptor,
+ GST_PAD_CAPS (cpad->collect.pad), cpad->mapping_data,
+ cpad->collect.buffer, source_package, source_track,
+ &track->edit_rate);
+
+ if (track->edit_rate.n != source_track->edit_rate.n ||
+ track->edit_rate.d != source_track->edit_rate.d) {
+ memcpy (&source_track->edit_rate, &track->edit_rate,
+ sizeof (MXFFraction));
+ }
+
+ if (track->edit_rate.d <= 0 || track->edit_rate.n <= 0) {
+ GST_ERROR_OBJECT (mux, "Invalid edit rate");
+ return GST_FLOW_ERROR;
+ }
+
+ if (min_edit_rate_d >
+ ((gdouble) track->edit_rate.n) / ((gdouble) track->edit_rate.d)) {
+ min_edit_rate_d =
+ ((gdouble) track->edit_rate.n) / ((gdouble) track->edit_rate.d);
+ memcpy (&min_edit_rate, &track->edit_rate, sizeof (MXFFraction));
+ }
+
+ sequence = track->parent.sequence = (MXFMetadataSequence *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SEQUENCE);
+ mxf_ul_set (&MXF_METADATA_BASE (sequence)->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (sequence)->instance_uid, sequence);
+
+ memcpy (&sequence->data_definition, &cpad->writer->data_definition,
+ 16);
+ sequence->n_structural_components = 1;
+ sequence->structural_components =
+ g_new0 (MXFMetadataStructuralComponent *, 1);
+
+ clip = (MXFMetadataSourceClip *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SOURCE_CLIP);
+ sequence->structural_components[0] =
+ (MXFMetadataStructuralComponent *) clip;
+ mxf_ul_set (&MXF_METADATA_BASE (clip)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (clip)->instance_uid, clip);
+
+ memcpy (&clip->parent.data_definition, &sequence->data_definition,
+ 16);
+ clip->start_position = 0;
+
+ memcpy (&clip->source_package_id, &cstorage->packages[1]->package_uid,
+ 32);
+ clip->source_track_id = n;
+
+ n++;
+ }
+
+ n = 0;
+ /* Timecode track */
+ {
+ MXFMetadataTimelineTrack *track;
+ MXFMetadataSequence *sequence;
+ MXFMetadataTimecodeComponent *component;
+
+ p->tracks[n] = (MXFMetadataTrack *)
+ gst_mini_object_new (MXF_TYPE_METADATA_TIMELINE_TRACK);
+ track = (MXFMetadataTimelineTrack *) p->tracks[n];
+ mxf_ul_set (&MXF_METADATA_BASE (track)->instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (track)->instance_uid, track);
+
+ track->parent.track_id = n + 1;
+ track->parent.track_number = 0;
+ track->parent.track_name = g_strdup ("Timecode track");
+ /* FIXME: Is this correct? */
+ memcpy (&track->edit_rate, &min_edit_rate, sizeof (MXFFraction));
+
+ sequence = track->parent.sequence = (MXFMetadataSequence *)
+ gst_mini_object_new (MXF_TYPE_METADATA_SEQUENCE);
+ mxf_ul_set (&MXF_METADATA_BASE (sequence)->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (sequence)->instance_uid, sequence);
+
+ memcpy (&sequence->data_definition,
+ mxf_metadata_track_identifier_get
+ (MXF_METADATA_TRACK_TIMECODE_12M_INACTIVE), 16);
+
+ sequence->n_structural_components = 1;
+ sequence->structural_components =
+ g_new0 (MXFMetadataStructuralComponent *, 1);
+
+ component = (MXFMetadataTimecodeComponent *)
+ gst_mini_object_new (MXF_TYPE_METADATA_TIMECODE_COMPONENT);
+ sequence->structural_components[0] =
+ (MXFMetadataStructuralComponent *) component;
+ mxf_ul_set (&MXF_METADATA_BASE (component)->instance_uid,
+ mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (component)->instance_uid, component);
+
+ memcpy (&component->parent.data_definition,
+ &sequence->data_definition, 16);
+
+ component->start_timecode = 0;
+ component->rounded_timecode_base =
+ (((gdouble) track->edit_rate.n) / ((gdouble) track->edit_rate.d) +
+ 0.5);
+ /* TODO: drop frame */
+ }
+
+ memcpy (&mux->min_edit_rate, &min_edit_rate, sizeof (MXFFraction));
+ }
+ }
+
+ for (i = 0; i < cstorage->packages[1]->n_tracks; i++) {
+ MXFMetadataTrack *track = cstorage->packages[1]->tracks[i];
+ guint j;
+ guint32 templ;
+ guint8 n_type, n;
+
+ if ((track->track_number & 0x00ff00ff) != 0)
+ continue;
+
+ templ = track->track_number;
+ n_type = 0;
+
+ for (j = 0; j < cstorage->packages[1]->n_tracks; j++) {
+ MXFMetadataTrack *tmp = cstorage->packages[1]->tracks[j];
+
+ if (tmp->track_number == templ) {
+ n_type++;
+ }
+ }
+
+ n = 0;
+ for (j = 0; j < cstorage->packages[1]->n_tracks; j++) {
+ MXFMetadataTrack *tmp = cstorage->packages[1]->tracks[j];
+
+ if (tmp->track_number == templ) {
+ n++;
+ tmp->track_number |= (n_type << 16) | (n);
+ }
+ }
+ }
+
+ cstorage->n_essence_container_data = 1;
+ cstorage->essence_container_data =
+ g_new0 (MXFMetadataEssenceContainerData *, 1);
+ cstorage->essence_container_data[0] = (MXFMetadataEssenceContainerData *)
+ gst_mini_object_new (MXF_TYPE_METADATA_ESSENCE_CONTAINER_DATA);
+ mxf_ul_set (&MXF_METADATA_BASE (cstorage->essence_container_data[0])->
+ instance_uid, mux->metadata);
+ g_hash_table_insert (mux->metadata,
+ &MXF_METADATA_BASE (cstorage->essence_container_data[0])->instance_uid,
+ cstorage->essence_container_data[0]);
+
+ cstorage->essence_container_data[0]->linked_package =
+ MXF_METADATA_SOURCE_PACKAGE (cstorage->packages[1]);
+ cstorage->essence_container_data[0]->index_sid = 0;
+ cstorage->essence_container_data[0]->body_sid = 1;
+ }
+
+ return ret;
+}
+
+static GstFlowReturn
+gst_mxf_mux_create_header_partition_pack (GstMXFMux * mux)
+{
+ GSList *l;
+ guint i = 0;
+
+ mxf_partition_pack_reset (&mux->partition);
+ mux->partition.type = MXF_PARTITION_PACK_HEADER;
+ mux->partition.closed = mux->partition.complete = FALSE;
+ mux->partition.major_version = 0x0001;
+ mux->partition.minor_version = 0x0002;
+ mux->partition.kag_size = 0;
+ mux->partition.this_partition = 0;
+ mux->partition.prev_partition = 0;
+ mux->partition.footer_partition = 0;
+ mux->partition.header_byte_count = 0;
+ mux->partition.index_byte_count = 0;
+ mux->partition.index_sid = 0;
+ mux->partition.body_offset = 0;
+ mux->partition.body_sid = 0;
+
+ memcpy (&mux->partition.operational_pattern,
+ &mux->preface->operational_pattern, 16);
+
+ mux->partition.n_essence_containers = g_slist_length (mux->collect->data);
+ mux->partition.essence_containers =
+ g_new0 (MXFUL, mux->partition.n_essence_containers);
+
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ guint j;
+ gboolean found = FALSE;
+
+ for (j = 0; j <= i; j++) {
+ if (mxf_ul_is_equal (&cpad->descriptor->essence_container,
+ &mux->partition.essence_containers[j])) {
+ found = TRUE;
+ break;
+ }
+ }
+
+ if (found)
+ continue;
+
+ memcpy (&mux->partition.essence_containers[i],
+ &cpad->descriptor->essence_container, 16);
+ i++;
+ }
+ mux->partition.n_essence_containers = i;
+
+ return GST_FLOW_OK;
+}
+
+static GstFlowReturn
+gst_mxf_mux_write_header_metadata (GstMXFMux * mux)
+{
+ GstFlowReturn ret = GST_FLOW_OK;
+ GstBuffer *buf;
+ GList *buffers = NULL;
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ GHashTableIter iter;
+#else
+ GList *values;
+#endif
+ MXFMetadataBase *m;
+ GList *l;
+ guint64 header_byte_count = 0;
+
+ buf =
+ mxf_metadata_base_to_buffer (MXF_METADATA_BASE (mux->preface),
+ &mux->primer);
+ header_byte_count += GST_BUFFER_SIZE (buf);
+ buffers = g_list_prepend (buffers, buf);
+
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ g_hash_table_iter_init (&iter, mux->metadata);
+ while (g_hash_table_iter_next (&iter, NULL, (gpointer) & m)) {
+#else
+ values = g_hash_table_get_values (mux->metadata);
+ for (l = values; l; l = l->next) {
+ m = l->data;
+#endif
+ buf = mxf_metadata_base_to_buffer (m, &mux->primer);
+ header_byte_count += GST_BUFFER_SIZE (buf);
+ buffers = g_list_prepend (buffers, buf);
+ }
+
+#if !GLIB_CHECK_VERSION (2, 16, 0)
+ g_list_free (value);
+#endif
+
+ buffers = g_list_reverse (buffers);
+ buf = mxf_primer_pack_to_buffer (&mux->primer);
+ header_byte_count += GST_BUFFER_SIZE (buf);
+ buffers = g_list_prepend (buffers, buf);
+
+ mux->partition.header_byte_count = header_byte_count;
+ buf = mxf_partition_pack_to_buffer (&mux->partition);
+ if ((ret = gst_mxf_mux_push (mux, buf)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (mux, "Failed pushing partition: %s",
+ gst_flow_get_name (ret));
+ g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
+ g_list_free (buffers);
+ return ret;
+ }
+
+ for (l = buffers; l; l = l->next) {
+ buf = l->data;
+ l->data = NULL;
+ if ((ret = gst_mxf_mux_push (mux, buf)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (mux, "Failed pushing buffer: %s",
+ gst_flow_get_name (ret));
+ g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
+ g_list_free (buffers);
+ return ret;
+ }
+ }
+
+ g_list_free (buffers);
+
+ return ret;
+}
+
+static const guint8 _gc_essence_element_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x01, 0x02, 0x01, 0x00,
+ 0x0d, 0x01, 0x03, 0x01, 0x00, 0x00, 0x00, 0x00
+};
+
+static GstFlowReturn
+gst_mxf_mux_handle_buffer (GstMXFMux * mux, GstMXFMuxPad * cpad)
+{
+ GstBuffer *buf = NULL;
+ GstBuffer *outbuf = NULL;
+ GstBuffer *packet;
+ GstFlowReturn ret = GST_FLOW_OK;
+ guint8 slen, ber[9];
+ gboolean flush =
+ (cpad->collect.abidata.ABI.eos && !cpad->have_complete_edit_unit
+ && cpad->collect.buffer == NULL);
+
+ if (cpad->have_complete_edit_unit) {
+ GST_DEBUG_OBJECT (cpad->collect.pad,
+ "Handling remaining buffer for track %u at position %" G_GINT64_FORMAT,
+ cpad->source_track->parent.track_id, cpad->pos);
+ buf = NULL;
+ } else if (!flush) {
+ buf = gst_collect_pads_pop (mux->collect, &cpad->collect);
+ }
+
+ if (buf) {
+ GST_DEBUG_OBJECT (cpad->collect.pad,
+ "Handling buffer of size %u for track %u at position %" G_GINT64_FORMAT,
+ GST_BUFFER_SIZE (buf), cpad->source_track->parent.track_id, cpad->pos);
+ } else {
+ flush = TRUE;
+ GST_DEBUG_OBJECT (cpad->collect.pad,
+ "Flushing for track %u at position %" G_GINT64_FORMAT,
+ cpad->source_track->parent.track_id, cpad->pos);
+ }
+
+ ret = cpad->write_func (buf, GST_PAD_CAPS (cpad->collect.pad),
+ cpad->mapping_data, cpad->adapter, &outbuf, flush);
+ if (ret != GST_FLOW_OK && ret != GST_FLOW_CUSTOM_SUCCESS) {
+ GST_ERROR_OBJECT (cpad->collect.pad,
+ "Failed handling buffer for track %u, reason %s",
+ cpad->source_track->parent.track_id, gst_flow_get_name (ret));
+ return ret;
+ }
+
+ if (ret == GST_FLOW_CUSTOM_SUCCESS) {
+ cpad->have_complete_edit_unit = TRUE;
+ ret = GST_FLOW_OK;
+ } else {
+ cpad->have_complete_edit_unit = FALSE;
+ }
+
+ buf = outbuf;
+ if (buf == NULL)
+ return ret;
+
+ slen = mxf_ber_encode_size (GST_BUFFER_SIZE (buf), ber);
+ packet = gst_buffer_new_and_alloc (16 + slen + GST_BUFFER_SIZE (buf));
+ memcpy (GST_BUFFER_DATA (packet), _gc_essence_element_ul, 16);
+ GST_BUFFER_DATA (packet)[7] = cpad->descriptor->essence_container.u[7];
+ GST_WRITE_UINT32_BE (&GST_BUFFER_DATA (packet)[12],
+ cpad->source_track->parent.track_number);
+ memcpy (&GST_BUFFER_DATA (packet)[16], ber, slen);
+ memcpy (&GST_BUFFER_DATA (packet)[16 + slen], GST_BUFFER_DATA (buf),
+ GST_BUFFER_SIZE (buf));
+ gst_buffer_unref (buf);
+
+ GST_DEBUG_OBJECT (cpad->collect.pad, "Pushing buffer of size %u for track %u",
+ GST_BUFFER_SIZE (packet), cpad->source_track->parent.track_id);
+
+ if ((ret = gst_mxf_mux_push (mux, packet)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (cpad->collect.pad,
+ "Failed pushing buffer for track %u, reason %s",
+ cpad->source_track->parent.track_id, gst_flow_get_name (ret));
+ return ret;
+ }
+
+ cpad->pos++;
+ cpad->last_timestamp =
+ gst_util_uint64_scale (GST_SECOND * cpad->pos,
+ cpad->source_track->edit_rate.d, cpad->source_track->edit_rate.n);
+
+ return ret;
+}
+
+static GstFlowReturn
+gst_mxf_mux_write_body_partition (GstMXFMux * mux)
+{
+ GstBuffer *buf;
+
+ mux->partition.type = MXF_PARTITION_PACK_BODY;
+ mux->partition.closed = mux->partition.complete = FALSE;
+ mux->partition.major_version = 0x0001;
+ mux->partition.minor_version = 0x0002;
+ mux->partition.kag_size = 0;
+ mux->partition.this_partition = mux->offset;
+ mux->partition.prev_partition = 0;
+ mux->partition.footer_partition = 0;
+ mux->partition.header_byte_count = 0;
+ mux->partition.index_byte_count = 0;
+ mux->partition.index_sid = 0;
+ mux->partition.body_offset = 0;
+ mux->partition.body_sid =
+ mux->preface->content_storage->essence_container_data[0]->body_sid;
+
+ buf = mxf_partition_pack_to_buffer (&mux->partition);
+ return gst_mxf_mux_push (mux, buf);
+}
+
+static GstFlowReturn
+gst_mxf_mux_handle_eos (GstMXFMux * mux)
+{
+ GSList *l;
+ gboolean have_data = FALSE;
+ GstBuffer *packet;
+
+ do {
+ GstMXFMuxPad *best = NULL;
+
+ have_data = FALSE;
+
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ GstClockTime next_gc_timestamp =
+ gst_util_uint64_scale ((mux->last_gc_position + 1) * GST_SECOND,
+ mux->min_edit_rate.d, mux->min_edit_rate.n);
+
+ best = NULL;
+
+ if (cpad->have_complete_edit_unit ||
+ gst_adapter_available (cpad->adapter) > 0 || cpad->collect.buffer) {
+ have_data = TRUE;
+ if (cpad->last_timestamp < next_gc_timestamp) {
+ best = cpad;
+ break;
+ }
+ } else if (have_data && !l->next) {
+ mux->last_gc_position++;
+ mux->last_gc_timestamp = next_gc_timestamp;
+ have_data = FALSE;
+ best = NULL;
+ break;
+ }
+ }
+
+ if (best) {
+ gst_mxf_mux_handle_buffer (mux, best);
+ have_data = TRUE;
+ }
+ } while (have_data);
+
+ mux->last_gc_position++;
+ mux->last_gc_timestamp =
+ gst_util_uint64_scale (mux->last_gc_position * GST_SECOND,
+ mux->min_edit_rate.d, mux->min_edit_rate.n);
+
+ /* Update essence track durations */
+ for (l = mux->collect->data; l; l = l->next) {
+ GstMXFMuxPad *cpad = l->data;
+ guint i;
+
+ /* Update durations */
+ cpad->source_track->parent.sequence->duration = cpad->pos;
+ MXF_METADATA_SOURCE_CLIP (cpad->source_track->parent.sequence->
+ structural_components[0])->parent.duration = cpad->pos;
+ for (i = 0; i < mux->preface->content_storage->packages[0]->n_tracks; i++) {
+ MXFMetadataTimelineTrack *track;
+
+ if (!MXF_IS_METADATA_TIMELINE_TRACK (mux->preface->content_storage->
+ packages[0]->tracks[i])
+ || !MXF_IS_METADATA_SOURCE_CLIP (mux->preface->content_storage->
+ packages[0]->tracks[i]->sequence->structural_components[0]))
+ continue;
+
+ track =
+ MXF_METADATA_TIMELINE_TRACK (mux->preface->content_storage->
+ packages[0]->tracks[i]);
+ if (MXF_METADATA_SOURCE_CLIP (track->parent.sequence->
+ structural_components[0])->source_track_id ==
+ cpad->source_track->parent.track_id) {
+ track->parent.sequence->structural_components[0]->duration = cpad->pos;
+ track->parent.sequence->duration = cpad->pos;
+ }
+ }
+ }
+
+ /* Update timecode track duration */
+ {
+ MXFMetadataTimelineTrack *track =
+ MXF_METADATA_TIMELINE_TRACK (mux->preface->content_storage->
+ packages[0]->tracks[0]);
+ MXFMetadataSequence *sequence = track->parent.sequence;
+ MXFMetadataTimecodeComponent *component =
+ MXF_METADATA_TIMECODE_COMPONENT (sequence->structural_components[0]);
+
+ sequence->duration = mux->last_gc_position;
+ component->parent.duration = mux->last_gc_position;
+ }
+
+ {
+ guint64 body_partition = mux->partition.this_partition;
+ guint32 body_sid = mux->partition.body_sid;
+ guint64 footer_partition = mux->offset;
+ GArray *rip;
+ GstFlowReturn ret;
+ MXFRandomIndexPackEntry entry;
+
+ mux->partition.type = MXF_PARTITION_PACK_FOOTER;
+ mux->partition.closed = TRUE;
+ mux->partition.complete = TRUE;
+ mux->partition.major_version = 0x0001;
+ mux->partition.minor_version = 0x0002;
+ mux->partition.kag_size = 0;
+ mux->partition.this_partition = mux->offset;
+ mux->partition.prev_partition = body_partition;
+ mux->partition.footer_partition = mux->offset;
+ mux->partition.header_byte_count = 0;
+ mux->partition.index_byte_count = 0;
+ mux->partition.index_sid = 0;
+ mux->partition.body_offset = 0;
+ mux->partition.body_sid = 0;
+
+ gst_mxf_mux_write_header_metadata (mux);
+
+ rip = g_array_sized_new (FALSE, FALSE, sizeof (MXFRandomIndexPackEntry), 3);
+ entry.offset = 0;
+ entry.body_sid = 0;
+ g_array_append_val (rip, entry);
+ entry.offset = body_partition;
+ entry.body_sid = body_sid;
+ g_array_append_val (rip, entry);
+ entry.offset = footer_partition;
+ entry.body_sid = 0;
+ g_array_append_val (rip, entry);
+
+ packet = mxf_random_index_pack_to_buffer (rip);
+ if ((ret = gst_mxf_mux_push (mux, packet)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (mux, "Failed pushing random index pack");
+ }
+ g_array_free (rip, TRUE);
+
+ /* Rewrite header partition with updated values */
+ if (gst_pad_push_event (mux->srcpad,
+ gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_BYTES, 0, -1,
+ 0))) {
+ mux->offset = 0;
+ mux->partition.type = MXF_PARTITION_PACK_HEADER;
+ mux->partition.closed = TRUE;
+ mux->partition.complete = TRUE;
+ mux->partition.major_version = 0x0001;
+ mux->partition.minor_version = 0x0002;
+ mux->partition.kag_size = 0;
+ mux->partition.this_partition = 0;
+ mux->partition.prev_partition = footer_partition;
+ mux->partition.footer_partition = footer_partition;
+ mux->partition.header_byte_count = 0;
+ mux->partition.index_byte_count = 0;
+ mux->partition.index_sid = 0;
+ mux->partition.body_offset = 0;
+ mux->partition.body_sid = 0;
+
+ ret = gst_mxf_mux_write_header_metadata (mux);
+ if (ret != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (mux, "Rewriting header partition failed");
+ return ret;
+ }
+ } else {
+ GST_WARNING_OBJECT (mux, "Can't rewrite header partition");
+ }
+ }
+
+ return GST_FLOW_OK;
+}
+
+static gint
+_sort_mux_pads (gconstpointer a, gconstpointer b)
+{
+ const GstMXFMuxPad *pa = a, *pb = b;
+ MXFMetadataTrackType ta =
+ mxf_metadata_track_identifier_parse (&pa->writer->data_definition);
+ MXFMetadataTrackType tb =
+ mxf_metadata_track_identifier_parse (&pb->writer->data_definition);
+
+ if (ta != tb)
+ return ta - tb;
+
+ return pa->source_track->parent.track_number -
+ pa->source_track->parent.track_number;
+}
+
+static GstFlowReturn
+gst_mxf_mux_collected (GstCollectPads * pads, gpointer user_data)
+{
+ GstMXFMux *mux = GST_MXF_MUX (user_data);
+ GstMXFMuxPad *best = NULL;
+ GstFlowReturn ret;
+ GSList *sl;
+ gboolean eos = TRUE;
+
+ if (mux->state == GST_MXF_MUX_STATE_ERROR) {
+ GST_ERROR_OBJECT (mux, "Had an error before -- returning");
+ return GST_FLOW_ERROR;
+ } else if (mux->state == GST_MXF_MUX_STATE_EOS) {
+ GST_WARNING_OBJECT (mux, "EOS");
+ return GST_FLOW_UNEXPECTED;
+ }
+
+ if (mux->state == GST_MXF_MUX_STATE_HEADER) {
+ if (mux->collect->data == NULL) {
+ GST_ELEMENT_ERROR (mux, STREAM, MUX, (NULL),
+ ("No input streams configured"));
+ ret = GST_FLOW_ERROR;
+ goto error;
+ }
+
+ if (gst_pad_push_event (mux->srcpad,
+ gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_BYTES, 0, -1,
+ 0))) {
+ if ((ret = gst_mxf_mux_create_metadata (mux)) != GST_FLOW_OK)
+ goto error;
+
+ if ((ret = gst_mxf_mux_create_header_partition_pack (mux)) != GST_FLOW_OK)
+ goto error;
+
+ ret = gst_mxf_mux_write_header_metadata (mux);
+ } else {
+ ret = GST_FLOW_ERROR;
+ }
+
+ if (ret != GST_FLOW_OK)
+ goto error;
+
+ /* Sort pads, we will always write in that order */
+ mux->collect->data = g_slist_sort (mux->collect->data, _sort_mux_pads);
+
+ /* Write body partition */
+ ret = gst_mxf_mux_write_body_partition (mux);
+ if (ret != GST_FLOW_OK)
+ goto error;
+ mux->state = GST_MXF_MUX_STATE_DATA;
+ }
+
+ g_return_val_if_fail (g_hash_table_size (mux->metadata) > 0, GST_FLOW_ERROR);
+
+ do {
+ for (sl = mux->collect->data; sl; sl = sl->next) {
+ GstMXFMuxPad *cpad = sl->data;
+ GstClockTime next_gc_timestamp =
+ gst_util_uint64_scale ((mux->last_gc_position + 1) * GST_SECOND,
+ mux->min_edit_rate.d, mux->min_edit_rate.n);
+
+ eos &= cpad->collect.abidata.ABI.eos;
+
+ if ((!cpad->collect.abidata.ABI.eos || cpad->have_complete_edit_unit ||
+ gst_adapter_available (cpad->adapter) > 0 || cpad->collect.buffer)
+ && cpad->last_timestamp < next_gc_timestamp) {
+ best = cpad;
+ break;
+ } else if (!eos && !sl->next) {
+ mux->last_gc_position++;
+ mux->last_gc_timestamp = next_gc_timestamp;
+ eos = FALSE;
+ best = NULL;
+ break;
+ }
+ }
+ } while (!eos && best == NULL);
+
+ if (!eos && best) {
+ ret = gst_mxf_mux_handle_buffer (mux, best);
+ if (ret != GST_FLOW_OK)
+ goto error;
+ } else if (eos) {
+ GST_DEBUG_OBJECT (mux, "Handling EOS");
+
+ gst_mxf_mux_handle_eos (mux);
+ gst_pad_push_event (mux->srcpad, gst_event_new_eos ());
+ mux->state = GST_MXF_MUX_STATE_EOS;
+ return GST_FLOW_UNEXPECTED;
+ }
+
+ return GST_FLOW_OK;
+
+error:
+ {
+ mux->state = GST_MXF_MUX_STATE_ERROR;
+ gst_pad_push_event (mux->srcpad, gst_event_new_eos ());
+ return ret;
+ }
+}
+
+static GstStateChangeReturn
+gst_mxf_mux_change_state (GstElement * element, GstStateChange transition)
+{
+ GstStateChangeReturn ret;
+ GstMXFMux *mux = GST_MXF_MUX (element);
+
+ switch (transition) {
+ case GST_STATE_CHANGE_NULL_TO_READY:
+ break;
+ case GST_STATE_CHANGE_READY_TO_PAUSED:
+ gst_collect_pads_start (mux->collect);
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ gst_collect_pads_stop (mux->collect);
+ break;
+ default:
+ break;
+ }
+
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+
+ switch (transition) {
+ case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ gst_mxf_mux_reset (mux);
+ break;
+ case GST_STATE_CHANGE_READY_TO_NULL:
+ break;
+ default:
+ break;
+ }
+
+ return ret;
+}
diff --git a/gst/mxf/mxfmux.h b/gst/mxf/mxfmux.h
new file mode 100644
index 00000000..81f232ad
--- /dev/null
+++ b/gst/mxf/mxfmux.h
@@ -0,0 +1,103 @@
+/* GStreamer
+ * Copyright (C) 2009 Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef __MXF_MUX_H__
+#define __MXF_MUX_H__
+
+#include <gst/gst.h>
+#include <gst/base/gstadapter.h>
+#include <gst/base/gstcollectpads.h>
+
+#include "mxfwrite.h"
+
+G_BEGIN_DECLS
+
+#define GST_TYPE_MXF_MUX \
+ (gst_mxf_mux_get_type ())
+#define GST_MXF_MUX(obj) \
+ (G_TYPE_CHECK_INSTANCE_CAST ((obj), GST_TYPE_MXF_MUX, GstMXFMux))
+#define GST_MXF_MUX_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_CAST ((klass), GST_TYPE_MXF_MUX, GstMXFMuxClass))
+#define GST_IS_MXF_MUX(obj) \
+ (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_MXF_MUX))
+#define GST_IS_MXF_MUX_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_TYPE ((klass), GST_TYPE_MXF_MUX))
+
+typedef struct
+{
+ GstCollectData collect;
+
+ guint64 pos;
+ GstClockTime last_timestamp;
+
+ MXFMetadataFileDescriptor *descriptor;
+
+ GstAdapter *adapter;
+ gboolean have_complete_edit_unit;
+
+ gpointer mapping_data;
+ const MXFEssenceElementWriter *writer;
+ MXFEssenceElementWriteFunc write_func;
+
+ MXFMetadataSourcePackage *source_package;
+ MXFMetadataTimelineTrack *source_track;
+} GstMXFMuxPad;
+
+typedef enum
+{
+ GST_MXF_MUX_STATE_HEADER,
+ GST_MXF_MUX_STATE_DATA,
+ GST_MXF_MUX_STATE_EOS,
+ GST_MXF_MUX_STATE_ERROR
+} GstMXFMuxState;
+
+typedef struct _GstMXFMux {
+ GstElement element;
+
+ GstPad *srcpad;
+ GstCollectPads *collect;
+ GstPadEventFunction collect_event;
+
+ GstMXFMuxState state;
+ guint n_pads;
+
+ guint64 offset;
+
+ MXFPartitionPack partition;
+ MXFPrimerPack primer;
+
+ GHashTable *metadata;
+ MXFMetadataPreface *preface;
+
+ MXFFraction min_edit_rate;
+ guint64 last_gc_position;
+ GstClockTime last_gc_timestamp;
+
+ gchar *application;
+} GstMXFMux;
+
+typedef struct _GstMXFMuxClass {
+ GstElementClass parent;
+} GstMXFMuxClass;
+
+GType gst_mxf_mux_get_type (void);
+
+G_END_DECLS
+
+#endif /* __MXF_MUX_H__ */
diff --git a/gst/mxf/mxfparse.c b/gst/mxf/mxfparse.c
index d4d52312..f0c0d0e5 100644
--- a/gst/mxf/mxfparse.c
+++ b/gst/mxf/mxfparse.c
@@ -29,61 +29,19 @@
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
-/* SMPTE 377M 3.3: A value of 0 for every field means unknown timestamp */
-static const MXFTimestamp mxf_timestamp_unknown = { 0, 0, 0, 0, 0, 0, 0 };
-
-static const MXFUMID umid_zero = { {0,} };
-static const MXFUL key_zero = { {0,} };
-
-/* UL common to all MXF UL */
-static const guint8 mxf_key[] = { 0x06, 0x0e, 0x2b, 0x34 };
-
-/* SMPTE 377M 6.1 */
-static const guint8 partition_pack_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01,
- 0x01
-};
-
-/* SMPTE 336M */
-static const guint8 fill_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x01, 0x01, 0x01, 0x01, 0x03, 0x01, 0x02, 0x10,
- 0x01, 0x00, 0x00, 0x00
-};
-
-/* SMPTE 377M 8.1 */
-static const guint8 primer_pack_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01,
- 0x01, 0x05, 0x01, 0x00
-};
-
-/* SMPTE 377M 8.6 */
-static const guint8 metadata_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x53, 0x01, 0x01, 0x0d, 0x01, 0x01, 0x01,
- 0x01
-};
-
-static const guint8 random_index_pack_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01,
- 0x01, 0x11, 0x01, 0x00
-};
-
-static const guint8 index_table_segment_key[] =
- { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x53, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01,
- 0x01, 0x10, 0x01, 0x00
-};
-
gboolean
-mxf_is_mxf_packet (const MXFUL * key)
+mxf_is_mxf_packet (const MXFUL * ul)
{
- return (memcmp (key, mxf_key, 4) == 0);
+ return mxf_ul_is_subclass (MXF_UL (SMPTE), ul);
}
/* SMPTE 377M 6.1: Check if this is a valid partition pack */
gboolean
-mxf_is_partition_pack (const MXFUL * key)
+mxf_is_partition_pack (const MXFUL * ul)
{
- if (memcmp (key, partition_pack_key, 13) == 0 && key->u[13] >= 0x02
- && key->u[13] <= 0x04 && key->u[14] < 0x05 && key->u[15] == 0x00)
+ if (mxf_ul_is_subclass (MXF_UL (PARTITION_PACK), ul) &&
+ ul->u[13] >= 0x02 && ul->u[13] <= 0x04 &&
+ ul->u[14] < 0x05 && ul->u[15] == 0x00)
return TRUE;
return FALSE;
@@ -91,10 +49,9 @@ mxf_is_partition_pack (const MXFUL * key)
/* SMPTE 377M 6.2: header partition pack has byte 14 == 0x02 */
gboolean
-mxf_is_header_partition_pack (const MXFUL * key)
+mxf_is_header_partition_pack (const MXFUL * ul)
{
- if (memcmp (key, partition_pack_key, 13) == 0 && key->u[13] == 0x02 &&
- key->u[14] < 0x05 && key->u[15] == 0x00)
+ if (mxf_is_partition_pack (ul) && ul->u[13] == 0x02)
return TRUE;
return FALSE;
@@ -102,10 +59,9 @@ mxf_is_header_partition_pack (const MXFUL * key)
/* SMPTE 377M 6.3: body partition pack has byte 14 == 0x03 */
gboolean
-mxf_is_body_partition_pack (const MXFUL * key)
+mxf_is_body_partition_pack (const MXFUL * ul)
{
- if (memcmp (key, partition_pack_key, 13) == 0 && key->u[13] == 0x03 &&
- key->u[14] < 0x05 && key->u[15] == 0x00)
+ if (mxf_is_partition_pack (ul) && ul->u[13] == 0x03)
return TRUE;
return FALSE;
@@ -113,159 +69,92 @@ mxf_is_body_partition_pack (const MXFUL * key)
/* SMPTE 377M 6.4: footer partition pack has byte 14 == 0x04 */
gboolean
-mxf_is_footer_partition_pack (const MXFUL * key)
+mxf_is_footer_partition_pack (const MXFUL * ul)
{
- if (memcmp (key, partition_pack_key, 13) == 0 && key->u[13] == 0x04 &&
- key->u[14] < 0x05 && key->u[15] == 0x00)
+ if (mxf_is_partition_pack (ul) && ul->u[13] == 0x04)
return TRUE;
return FALSE;
}
gboolean
-mxf_is_fill (const MXFUL * key)
+mxf_is_fill (const MXFUL * ul)
{
- return (memcmp (key, fill_key, 16) == 0);
+ return (mxf_ul_is_subclass (MXF_UL (FILL), ul));
}
gboolean
-mxf_is_primer_pack (const MXFUL * key)
+mxf_is_primer_pack (const MXFUL * ul)
{
- return (memcmp (key, primer_pack_key, 16) == 0);
+ return (mxf_ul_is_subclass (MXF_UL (PRIMER_PACK), ul));
}
gboolean
-mxf_is_metadata (const MXFUL * key)
+mxf_is_metadata (const MXFUL * ul)
{
- return (memcmp (key, metadata_key, 13) == 0 && key->u[15] == 0x00);
+ return (mxf_ul_is_subclass (MXF_UL (METADATA), ul));
}
/* SMPTE 377M 8.7.3 */
gboolean
-mxf_is_descriptive_metadata (const MXFUL * key)
+mxf_is_descriptive_metadata (const MXFUL * ul)
{
- return (memcmp (key, mxf_key, 4) == 0 &&
- key->u[4] == 0x02 &&
- key->u[6] == 0x01 &&
- key->u[7] == 0x01 &&
- key->u[8] == 0x0d &&
- key->u[9] == 0x01 && key->u[10] == 0x04 && key->u[11] == 0x01);
+ return (mxf_ul_is_subclass (MXF_UL (DESCRIPTIVE_METADATA), ul));
}
gboolean
-mxf_is_random_index_pack (const MXFUL * key)
+mxf_is_random_index_pack (const MXFUL * ul)
{
- return (memcmp (key, random_index_pack_key, 16) == 0);
+ return (mxf_ul_is_subclass (MXF_UL (RANDOM_INDEX_PACK), ul));
}
gboolean
-mxf_is_index_table_segment (const MXFUL * key)
+mxf_is_index_table_segment (const MXFUL * ul)
{
- return (memcmp (key, index_table_segment_key, 16) == 0);
+ return (mxf_ul_is_subclass (MXF_UL (INDEX_TABLE_SEGMENT), ul));
}
/* SMPTE 379M 6.2.1 */
gboolean
-mxf_is_generic_container_system_item (const MXFUL * key)
+mxf_is_generic_container_system_item (const MXFUL * ul)
{
- return (memcmp (key, mxf_key, 4) == 0 && key->u[4] == 0x02
- && key->u[6] == 0x01 && key->u[8] == 0x0d && key->u[9] == 0x01
- && key->u[10] == 0x03 && key->u[11] == 0x01 && (key->u[12] == 0x04
- || key->u[12] == 0x14));
+ return (mxf_ul_is_subclass (MXF_UL (GENERIC_CONTAINER_SYSTEM_ITEM), ul) &&
+ (ul->u[12] == 0x04 || ul->u[12] == 0x14));
}
/* SMPTE 379M 7.1 */
gboolean
-mxf_is_generic_container_essence_element (const MXFUL * key)
+mxf_is_generic_container_essence_element (const MXFUL * ul)
{
- return (memcmp (key, mxf_key, 4) == 0 && key->u[4] == 0x01
- && key->u[5] == 0x02 && key->u[6] == 0x01 && key->u[8] == 0x0d
- && key->u[9] == 0x01 && key->u[10] == 0x03 && key->u[11] == 0x01
- && (key->u[12] == 0x05 || key->u[12] == 0x06 || key->u[12] == 0x07
- || key->u[12] == 0x15 || key->u[12] == 0x16 || key->u[12] == 0x17
- || key->u[12] == 0x18));
+ return (mxf_ul_is_subclass (MXF_UL (GENERIC_CONTAINER_ESSENCE_ELEMENT), ul)
+ && (ul->u[12] == 0x05 || ul->u[12] == 0x06
+ || ul->u[12] == 0x07 || ul->u[12] == 0x15
+ || ul->u[12] == 0x16 || ul->u[12] == 0x17 || ul->u[12] == 0x18));
}
/* SMPTE 379M 8 */
gboolean
-mxf_is_generic_container_essence_container_label (const MXFUL * key)
+mxf_is_generic_container_essence_container_label (const MXFUL * ul)
{
- return (key->u[0] == 0x06 &&
- key->u[1] == 0x0e &&
- key->u[2] == 0x2b &&
- key->u[3] == 0x34 &&
- key->u[4] == 0x04 &&
- key->u[5] == 0x01 &&
- key->u[6] == 0x01 &&
- key->u[8] == 0x0d &&
- key->u[9] == 0x01 &&
- key->u[10] == 0x03 &&
- key->u[11] == 0x01 && (key->u[12] == 0x01 || key->u[12] == 0x02));
+ return (mxf_ul_is_subclass (MXF_UL
+ (GENERIC_CONTAINER_ESSENCE_CONTAINER_LABEL), ul) && (ul->u[12] == 0x01
+ || ul->u[12] == 0x02));
}
/* Essence container label found in files generated by Avid */
-static const guint8 avid_essence_container_label[] = {
- 0x06, 0x0e, 0x2b, 0x34, 0x01, 0x01, 0x01, 0xff, 0x4b, 0x46, 0x41, 0x41, 0x00,
- 0x0d, 0x4d, 0x4f
-};
-
gboolean
-mxf_is_avid_essence_container_label (const MXFUL * key)
+mxf_is_avid_essence_container_label (const MXFUL * ul)
{
- return (memcmp (&key->u, avid_essence_container_label, 16) == 0);
+ return (mxf_ul_is_subclass (MXF_UL (AVID_ESSENCE_CONTAINER_ESSENCE_LABEL),
+ ul));
}
/* Essence element key found in files generated by Avid */
-static const guint8 avid_essence_element_ul[] = {
- 0x06, 0x0e, 0x2b, 0x34, 0x01, 0x02, 0x01, 0x01, 0x0e, 0x04, 0x03, 0x01, 0x00,
- 0x00, 0x00, 0x00
-};
-
gboolean
-mxf_is_avid_essence_container_essence_element (const MXFUL * key)
+mxf_is_avid_essence_container_essence_element (const MXFUL * ul)
{
- return (memcmp (&key->u, avid_essence_element_ul, 12) == 0);
-}
-
-gboolean
-mxf_ul_is_equal (const MXFUL * a, const MXFUL * b)
-{
- return (memcmp (a, b, 16) == 0);
-}
-
-gboolean
-mxf_ul_is_zero (const MXFUL * key)
-{
- return (memcmp (key, &key_zero, 16) == 0);
-}
-
-guint
-mxf_ul_hash (const MXFUL * key)
-{
- guint32 ret = 0;
- guint i;
-
- for (i = 0; i < 4; i++)
- ret ^=
- (key->u[i * 4 + 0] << 24) | (key->u[i * 4 + 1] << 16) | (key->u[i * 4 +
- 2] << 8) | (key->u[i * 4 + 3] << 0);
-
- return ret;
-}
-
-gchar *
-mxf_ul_to_string (const MXFUL * key, gchar str[48])
-{
- g_return_val_if_fail (key != NULL, NULL);
- g_return_val_if_fail (str != NULL, NULL);
-
- g_snprintf (str, 48,
- "%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x.%02x",
- key->u[0], key->u[1], key->u[2], key->u[3], key->u[4], key->u[5],
- key->u[6], key->u[7], key->u[8], key->u[9], key->u[10], key->u[11],
- key->u[12], key->u[13], key->u[14], key->u[15]);
-
- return str;
+ return (mxf_ul_is_subclass (MXF_UL (AVID_ESSENCE_CONTAINER_ESSENCE_ELEMENT),
+ ul));
}
gboolean
@@ -277,7 +166,9 @@ mxf_umid_is_equal (const MXFUMID * a, const MXFUMID * b)
gboolean
mxf_umid_is_zero (const MXFUMID * umid)
{
- return (memcmp (umid, &umid_zero, 32) == 0);
+ static const MXFUMID zero = { {0,} };
+
+ return (memcmp (umid, &zero, 32) == 0);
}
gchar *
@@ -362,10 +253,13 @@ mxf_timestamp_parse (MXFTimestamp * timestamp, const guint8 * data, guint size)
return TRUE;
}
+/* SMPTE 377M 3.3: A value of 0 for every field means unknown timestamp */
gboolean
mxf_timestamp_is_unknown (const MXFTimestamp * a)
{
- return (memcmp (a, &mxf_timestamp_unknown, sizeof (MXFTimestamp)) == 0);
+ static const MXFTimestamp unknown = { 0, 0, 0, 0, 0, 0, 0 };
+
+ return (memcmp (a, &unknown, sizeof (MXFTimestamp)) == 0);
}
gint
@@ -469,54 +363,11 @@ mxf_product_version_parse (MXFProductVersion * product_version,
}
gboolean
-mxf_ul_array_parse (MXFUL ** array, guint32 * count, const guint8 * data,
- guint size)
+mxf_product_version_is_valid (const MXFProductVersion * version)
{
- guint32 element_count, element_size;
- guint i;
-
- g_return_val_if_fail (array != NULL, FALSE);
- g_return_val_if_fail (count != NULL, FALSE);
- g_return_val_if_fail (data != NULL, FALSE);
+ static const guint8 null[sizeof (MXFProductVersion)] = { 0, };
- if (size < 8)
- return FALSE;
-
- element_count = GST_READ_UINT32_BE (data);
- data += 4;
- size -= 4;
-
- if (element_count == 0) {
- *array = NULL;
- *count = 0;
- return TRUE;
- }
-
- element_size = GST_READ_UINT32_BE (data);
- data += 4;
- size -= 4;
-
- if (element_size != 16) {
- *array = NULL;
- *count = 0;
- return FALSE;
- }
-
- if (16 * element_count < size) {
- *array = NULL;
- *count = 0;
- return FALSE;
- }
-
- *array = g_new (MXFUL, element_count);
- *count = element_count;
-
- for (i = 0; i < element_count; i++) {
- memcpy (&((*array)[i]), data, 16);
- data += 16;
- }
-
- return TRUE;
+ return (memcmp (version, &null, sizeof (MXFProductVersion)) == 0);
}
/* SMPTE 377M 6.1, Table 2 */
@@ -1023,7 +874,12 @@ mxf_primer_pack_reset (MXFPrimerPack * pack)
if (pack->mappings)
g_hash_table_destroy (pack->mappings);
+ if (pack->reverse_mappings)
+ g_hash_table_destroy (pack->reverse_mappings);
+
memset (pack, 0, sizeof (MXFPrimerPack));
+
+ pack->next_free_tag = 0x8000;
}
/* structural metadata parsing */
@@ -1051,7 +907,10 @@ mxf_local_tag_parse (const guint8 * data, guint size, guint16 * tag,
void
mxf_local_tag_free (MXFLocalTag * tag)
{
- g_free (tag->data);
+ if (tag->g_slice)
+ g_slice_free1 (tag->size, tag->data);
+ else
+ g_free (tag->data);
g_slice_free (MXFLocalTag, tag);
}
@@ -1100,6 +959,32 @@ mxf_local_tag_add_to_hash_table (const MXFPrimerPack * primer,
return TRUE;
}
+gboolean
+mxf_local_tag_insert (MXFLocalTag * tag, GHashTable ** hash_table)
+{
+#ifndef GST_DISABLE_GST_DEBUG
+ gchar str[48];
+#endif
+
+ g_return_val_if_fail (tag != NULL, FALSE);
+ g_return_val_if_fail (hash_table != NULL, FALSE);
+
+ if (*hash_table == NULL)
+ *hash_table =
+ g_hash_table_new_full ((GHashFunc) mxf_ul_hash,
+ (GEqualFunc) mxf_ul_is_equal, (GDestroyNotify) NULL,
+ (GDestroyNotify) mxf_local_tag_free);
+
+ g_return_val_if_fail (*hash_table != NULL, FALSE);
+
+ GST_DEBUG ("Adding local tag 0x%04x with UL %s and size %u", tag,
+ mxf_ul_to_string (&tag->key, str), tag->size);
+
+ g_hash_table_insert (*hash_table, &tag->key, tag);
+
+ return TRUE;
+}
+
static GSList *_mxf_essence_element_handler_registry = NULL;
void
diff --git a/gst/mxf/mxfparse.h b/gst/mxf/mxfparse.h
index 5a5e630e..58d88d6b 100644
--- a/gst/mxf/mxfparse.h
+++ b/gst/mxf/mxfparse.h
@@ -25,6 +25,7 @@
#include <string.h>
#include "mxftypes.h"
+#include "mxful.h"
#include "mxfmetadata.h"
typedef GstFlowReturn (*MXFEssenceElementHandleFunc) (const MXFUL *key, GstBuffer *buffer, GstCaps *caps, MXFMetadataTimelineTrack *track, gpointer mapping_data, GstBuffer **outbuf);
@@ -34,11 +35,6 @@ typedef struct {
GstCaps * (*create_caps) (MXFMetadataTimelineTrack *track, GstTagList **tags, MXFEssenceElementHandleFunc *handler, gpointer *mapping_data);
} MXFEssenceElementHandler;
-gchar * mxf_ul_to_string (const MXFUL *ul, gchar str[48]);
-gboolean mxf_ul_is_equal (const MXFUL *a, const MXFUL *b);
-gboolean mxf_ul_is_zero (const MXFUL *ul);
-guint mxf_ul_hash (const MXFUL *ul);
-
gchar *mxf_umid_to_string (const MXFUMID * umid, gchar str[96]);
MXFUMID *mxf_umid_from_string (const gchar *str, MXFUMID * umid);
gboolean mxf_umid_is_equal (const MXFUMID *a, const MXFUMID *b);
@@ -72,6 +68,7 @@ gchar * mxf_utf16_to_utf8 (const guint8 * data, guint size);
gboolean mxf_product_version_parse (MXFProductVersion * product_version,
const guint8 * data, guint size);
+gboolean mxf_product_version_is_valid (const MXFProductVersion *version);
gboolean mxf_fraction_parse (MXFFraction *fraction, const guint8 *data, guint size);
gdouble mxf_fraction_to_double (const MXFFraction *fraction);
@@ -81,8 +78,6 @@ gboolean mxf_timestamp_is_unknown (const MXFTimestamp *a);
gint mxf_timestamp_compare (const MXFTimestamp *a, const MXFTimestamp *b);
gchar *mxf_timestamp_to_string (const MXFTimestamp *t, gchar str[32]);
-gboolean mxf_ul_array_parse (MXFUL **array, guint32 *count, const guint8 *data, guint size);
-
gboolean mxf_partition_pack_parse (const MXFUL *key, MXFPartitionPack *pack, const guint8 *data, guint size);
void mxf_partition_pack_reset (MXFPartitionPack *pack);
@@ -96,11 +91,12 @@ void mxf_index_table_segment_reset (MXFIndexTableSegment *segment);
gboolean mxf_local_tag_parse (const guint8 * data, guint size, guint16 * tag,
guint16 * tag_size, const guint8 ** tag_data);
-void gst_mxf_local_tag_free (MXFLocalTag *tag);
+void mxf_local_tag_free (MXFLocalTag *tag);
gboolean mxf_local_tag_add_to_hash_table (const MXFPrimerPack *primer,
guint16 tag, const guint8 *tag_data, guint16 tag_size,
GHashTable **hash_table);
+gboolean mxf_local_tag_insert (MXFLocalTag *tag, GHashTable **hash_table);
void mxf_essence_element_handler_register (const MXFEssenceElementHandler *handler);
const MXFEssenceElementHandler * mxf_essence_element_handler_find (const MXFMetadataTimelineTrack *track);
diff --git a/gst/mxf/mxftypes.h b/gst/mxf/mxftypes.h
index dcbbd28d..ffb40d9c 100644
--- a/gst/mxf/mxftypes.h
+++ b/gst/mxf/mxftypes.h
@@ -29,6 +29,10 @@ typedef struct {
guint8 u[16];
} MXFUL;
+typedef struct {
+ guint8 u[16];
+} MXFUUID;
+
/* SMPTE 377M 3.2 */
typedef struct {
guint8 u[32];
@@ -65,6 +69,8 @@ typedef struct {
MXFUL key;
guint16 size;
guint8 *data;
+
+ gboolean g_slice; /* TRUE if data was allocated by GSlice */
} MXFLocalTag;
/* SMPTE 377M 11.1 */
@@ -114,6 +120,8 @@ typedef struct {
typedef struct {
guint64 offset;
GHashTable *mappings;
+ GHashTable *reverse_mappings;
+ guint16 next_free_tag;
} MXFPrimerPack;
/* SMPTE 377M 10.2.3 */
diff --git a/gst/mxf/mxful.c b/gst/mxf/mxful.c
new file mode 100644
index 00000000..37624c53
--- /dev/null
+++ b/gst/mxf/mxful.c
@@ -0,0 +1,268 @@
+/* GStreamer
+ * Copyright (C) <2009> Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <string.h>
+
+#include "mxful.h"
+
+const MXFUL _mxf_ul_table[] = {
+ /* SMPTE */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x00,}},
+ /* FILL, SMPTE 336M */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x03, 0x01, 0x02, 0x10, 0x01, 0x00,}},
+ /* PARTITION_PACK, SMPTE 377M 6.1 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01, 0x00,}},
+ /* PRIMER_PACK, SMPTE 377M 8.1 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01, 0x05, 0x01, 0x00}},
+ /* METADATA, SMPTE 377M 8.6 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x53, 0x01, 0x01,
+ 0x0d, 0x01, 0x01, 0x01, 0x01, 0x00,}},
+ /* DESCRIPTIVE_METADATA, SMPTE 377M 8.7.3 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x00, 0x01, 0x01,
+ 0x0d, 0x01, 0x04, 0x01, 0x00,}},
+ /* RANDOM_INDEX_PACK, SMPTE 377M 11.1 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01, 0x11, 0x01, 0x00}},
+ /* INDEX_TABLE_SEGMENT, SMPTE 377M 10.2.2 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x53, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01, 0x10, 0x01, 0x00}},
+ /* GENERIC_CONTAINER_SYSTEM_ITEM, SMPTE 379M 6.2.1 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x02, 0x00, 0x01, 0x00,
+ 0x0d, 0x01, 0x03, 0x01, 0x00}},
+ /* GENERIC_CONTAINER_ESSENCE_ELEMENT, SMPTE 379M 7.1 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x01, 0x02, 0x01, 0x00,
+ 0x0d, 0x01, 0x03, 0x01, 0x00,}},
+ /* GENERIC_CONTAINER_ESSENCE_CONTAINER_LABEL, SMPTE 379M 8 */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x00,
+ 0x0d, 0x01, 0x03, 0x01, 0x00,}},
+ /* AVID_ESSENCE_CONTAINER_ESSENCE_ELEMENT, undocumented */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x01, 0x02, 0x01, 0x01,
+ 0x0e, 0x04, 0x03, 0x01, 0x00,}},
+ /* AVID_ESSENCE_CONTAINER_ESSENCE_LABEL, undocumented */
+ {{0x06, 0x0e, 0x2b, 0x34, 0x01, 0x01, 0x01, 0xff,
+ 0x4b, 0x46, 0x41, 0x41, 0x00, 0x0d, 0x4d, 0x4f}},
+};
+
+gboolean
+mxf_ul_is_equal (const MXFUL * a, const MXFUL * b)
+{
+ guint i;
+
+ g_return_val_if_fail (a != NULL, FALSE);
+ g_return_val_if_fail (b != NULL, FALSE);
+
+ for (i = 0; i < 16; i++) {
+ /* registry version */
+ if (i == 7)
+ continue;
+
+ if (a->u[i] != b->u[i])
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+gboolean
+mxf_ul_is_subclass (const MXFUL * class, const MXFUL * subclass)
+{
+ guint i;
+
+ g_return_val_if_fail (class != NULL, FALSE);
+ g_return_val_if_fail (subclass != NULL, FALSE);
+
+ for (i = 0; i < 16; i++) {
+ if (i == 7)
+ /* registry version */
+ continue;
+
+ if (class->u[i] != 0x00 && class->u[i] != subclass->u[i])
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+gboolean
+mxf_ul_is_zero (const MXFUL * ul)
+{
+ static const guint8 zero[16] = { 0, };
+
+ g_return_val_if_fail (ul != NULL, FALSE);
+
+ return (memcmp (ul, &zero, 16) == 0);
+}
+
+gboolean
+mxf_ul_is_valid (const MXFUL * ul)
+{
+ guint i, j;
+
+ g_return_val_if_fail (ul != NULL, FALSE);
+
+ for (i = 0; i < 16; i++) {
+ if (ul->u[i] == 0x00) {
+ for (j = i; j < 16; j++) {
+ if (ul->u[j] != 0x00)
+ return FALSE;
+ }
+
+ return TRUE;
+ }
+
+ if (ul->u[i] > 0x7f)
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+guint
+mxf_ul_hash (const MXFUL * ul)
+{
+ guint32 ret = 0;
+ guint i;
+
+ g_return_val_if_fail (ul != NULL, 0);
+
+ for (i = 0; i < 4; i++)
+ ret ^= (ul->u[i * 4 + 0] << 24) |
+ (ul->u[i * 4 + 1] << 16) |
+ (ul->u[i * 4 + 2] << 8) | (ul->u[i * 4 + 3] << 0);
+
+ return ret;
+}
+
+gchar *
+mxf_ul_to_string (const MXFUL * ul, gchar str[48])
+{
+ gchar *ret = str;
+
+ g_return_val_if_fail (ul != NULL, NULL);
+
+ if (ret == NULL)
+ ret = g_malloc (48);
+
+ g_snprintf (ret, 48,
+ "%02x.%02x.%02x.%02x."
+ "%02x.%02x.%02x.%02x."
+ "%02x.%02x.%02x.%02x."
+ "%02x.%02x.%02x.%02x",
+ ul->u[0], ul->u[1], ul->u[2], ul->u[3],
+ ul->u[4], ul->u[5], ul->u[6], ul->u[7],
+ ul->u[8], ul->u[9], ul->u[10], ul->u[11],
+ ul->u[12], ul->u[13], ul->u[14], ul->u[15]);
+
+ return ret;
+}
+
+MXFUL *
+mxf_ul_from_string (const gchar * str, MXFUL * ul)
+{
+ MXFUL *ret = ul;
+ gint len;
+ guint i, j;
+
+ g_return_val_if_fail (str != NULL, NULL);
+
+ len = strlen (str);
+ if (len != 47) {
+ GST_ERROR ("Invalid UL string length %d, should be 47", len);
+ return NULL;
+ }
+
+ if (ret == NULL)
+ ret = g_new0 (MXFUL, 1);
+
+ memset (ret, 0, 16);
+
+ for (i = 0, j = 0; i < 16; i++) {
+ if (!g_ascii_isxdigit (str[j]) ||
+ !g_ascii_isxdigit (str[j + 1]) ||
+ (str[j + 2] != '.' && str[j + 2] != '\0')) {
+ GST_ERROR ("Invalid UL string '%s'", str);
+ if (ul == NULL)
+ g_free (ret);
+ return NULL;
+ }
+
+ ret->u[i] = (g_ascii_xdigit_value (str[j]) << 4) |
+ (g_ascii_xdigit_value (str[j + 1]));
+ j += 3;
+ }
+ return ret;
+}
+
+gboolean
+mxf_ul_array_parse (MXFUL ** array, guint32 * count, const guint8 * data,
+ guint size)
+{
+ guint32 element_count, element_size;
+ guint i;
+
+ g_return_val_if_fail (array != NULL, FALSE);
+ g_return_val_if_fail (count != NULL, FALSE);
+ g_return_val_if_fail (data != NULL, FALSE);
+
+ if (size < 8)
+ return FALSE;
+
+ element_count = GST_READ_UINT32_BE (data);
+ data += 4;
+ size -= 4;
+
+ if (element_count == 0) {
+ *array = NULL;
+ *count = 0;
+ return TRUE;
+ }
+
+ element_size = GST_READ_UINT32_BE (data);
+ data += 4;
+ size -= 4;
+
+ if (element_size != 16) {
+ *array = NULL;
+ *count = 0;
+ return FALSE;
+ }
+
+ if (16 * element_count < size) {
+ *array = NULL;
+ *count = 0;
+ return FALSE;
+ }
+
+ *array = g_new (MXFUL, element_count);
+ *count = element_count;
+
+ for (i = 0; i < element_count; i++) {
+ memcpy (&((*array)[i]), data, 16);
+ data += 16;
+ }
+
+ return TRUE;
+}
diff --git a/gst/mxf/mxful.h b/gst/mxf/mxful.h
new file mode 100644
index 00000000..b187229d
--- /dev/null
+++ b/gst/mxf/mxful.h
@@ -0,0 +1,59 @@
+/* GStreamer
+ * Copyright (C) <2009> Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef __MXF_UL_H__
+#define __MXF_UL_H__
+
+#include <gst/gst.h>
+
+#include "mxftypes.h"
+
+typedef enum {
+ MXF_UL_SMPTE = 0,
+ MXF_UL_FILL,
+ MXF_UL_PARTITION_PACK,
+ MXF_UL_PRIMER_PACK,
+ MXF_UL_METADATA,
+ MXF_UL_DESCRIPTIVE_METADATA,
+ MXF_UL_RANDOM_INDEX_PACK,
+ MXF_UL_INDEX_TABLE_SEGMENT,
+ MXF_UL_GENERIC_CONTAINER_SYSTEM_ITEM,
+ MXF_UL_GENERIC_CONTAINER_ESSENCE_ELEMENT,
+ MXF_UL_GENERIC_CONTAINER_ESSENCE_CONTAINER_LABEL,
+ MXF_UL_AVID_ESSENCE_CONTAINER_ESSENCE_ELEMENT,
+ MXF_UL_AVID_ESSENCE_CONTAINER_ESSENCE_LABEL,
+ MXF_UL_MAX
+} MXFULId;
+
+extern const MXFUL _mxf_ul_table[MXF_UL_MAX];
+
+#define MXF_UL(id) (&_mxf_ul_table[MXF_UL_##id])
+
+gboolean mxf_ul_is_equal (const MXFUL *a, const MXFUL *b);
+gboolean mxf_ul_is_subclass (const MXFUL *class, const MXFUL *subclass);
+gboolean mxf_ul_is_zero (const MXFUL *ul);
+gboolean mxf_ul_is_valid (const MXFUL *ul);
+guint mxf_ul_hash (const MXFUL *ul);
+
+gchar * mxf_ul_to_string (const MXFUL *ul, gchar str[48]);
+MXFUL * mxf_ul_from_string (const gchar *str, MXFUL *ul);
+
+gboolean mxf_ul_array_parse (MXFUL **array, guint32 *count, const guint8 *data, guint size);
+
+#endif /* __MXF_UL_H__ */
diff --git a/gst/mxf/mxfup.c b/gst/mxf/mxfup.c
index ccc34ffd..a86a2c19 100644
--- a/gst/mxf/mxfup.c
+++ b/gst/mxf/mxfup.c
@@ -24,7 +24,6 @@
/* TODO:
* - Handle CDCI essence
* - Handle more formats with RGBA descriptor (4:4:4 / 4:4:4:4 YUV, RGB656, ...)
- * - Correctly transform for the GStreamer strides
* - Handle all the dimensions and other properties in the picture
* essence descriptors correctly according to S377M Annex E
* - Handle interlaced correctly, i.e. weave until we support one-field-per-buffer
@@ -40,12 +39,65 @@
#include <gst/video/video.h>
#include "mxfup.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
+static const struct
+{
+ const gchar *caps;
+ guint32 n_pixel_layout;
+ guint8 pixel_layout[10];
+ guint32 fourcc;
+} _rgba_mapping_table[] = {
+ {
+ GST_VIDEO_CAPS_RGB, 3, {
+ 'R', 8, 'G', 8, 'B', 8}, GST_MAKE_FOURCC ('R', 'G', 'B', ' ')}, {
+ GST_VIDEO_CAPS_BGR, 3, {
+ 'B', 8, 'G', 8, 'R', 8}, GST_MAKE_FOURCC ('B', 'G', 'R', ' ')}, {
+ GST_VIDEO_CAPS_YUV ("v308"), 3, {
+ 'Y', 8, 'U', 8, 'V', 8}, GST_MAKE_FOURCC ('v', '3', '0', '8')}, {
+ GST_VIDEO_CAPS_xRGB, 4, {
+ 'F', 8, 'R', 8, 'G', 8, 'B', 8}, GST_MAKE_FOURCC ('x', 'R', 'G', 'B')}, {
+ GST_VIDEO_CAPS_RGBx, 4, {
+ 'R', 8, 'G', 8, 'B', 8, 'F', 8}, GST_MAKE_FOURCC ('R', 'G', 'B', 'x')}, {
+ GST_VIDEO_CAPS_xBGR, 4, {
+ 'F', 8, 'B', 8, 'G', 8, 'R', 8}, GST_MAKE_FOURCC ('x', 'B', 'G', 'R')}, {
+ GST_VIDEO_CAPS_BGRx, 4, {
+ 'B', 8, 'G', 8, 'R', 8, 'F', 8}, GST_MAKE_FOURCC ('B', 'G', 'R', 'x')}, {
+ GST_VIDEO_CAPS_RGBA, 4, {
+ 'R', 8, 'G', 8, 'B', 8, 'A', 8}, GST_MAKE_FOURCC ('R', 'G', 'B', 'A')}, {
+ GST_VIDEO_CAPS_ARGB, 4, {
+ 'A', 8, 'R', 8, 'G', 8, 'B', 8}, GST_MAKE_FOURCC ('A', 'R', 'G', 'B')}, {
+ GST_VIDEO_CAPS_BGRA, 4, {
+ 'B', 8, 'G', 8, 'R', 8, 'A', 8}, GST_MAKE_FOURCC ('B', 'G', 'R', 'A')}, {
+ GST_VIDEO_CAPS_ABGR, 4, {
+ 'A', 8, 'B', 8, 'G', 8, 'R', 8}, GST_MAKE_FOURCC ('A', 'B', 'G', 'R')}, {
+ GST_VIDEO_CAPS_YUV ("AYUV"), 4, {
+ 'A', 8, 'Y', 8, 'U', 8, 'V', 8}, GST_MAKE_FOURCC ('A', 'Y', 'U', 'V')}
+};
+
+static const struct
+{
+ const gchar *caps;
+ guint bpp;
+ guint horizontal_subsampling;
+ guint vertical_subsampling;
+ gboolean reversed_byte_order;
+ guint32 fourcc;
+} _cdci_mapping_table[] = {
+ {
+ GST_VIDEO_CAPS_YUV ("YUY2"), 2, 1, 0, TRUE, GST_MAKE_FOURCC ('Y', 'U', 'Y',
+ '2')}, {
+GST_VIDEO_CAPS_YUV ("UYVY"), 2, 1, 0, FALSE, GST_MAKE_FOURCC ('U', 'Y', 'V',
+ 'Y')},};
+
typedef struct
{
+ guint32 fourcc; /* fourcc or RGB format specifier */
+ gint width, height;
+ guint bpp;
guint32 image_start_offset;
guint32 image_end_offset;
} MXFUPMappingData;
@@ -93,7 +145,6 @@ mxf_up_handle_essence_element (const MXFUL * key, GstBuffer * buffer,
}
if (!data || (data->image_start_offset == 0 && data->image_end_offset == 0)) {
- *outbuf = buffer;
} else {
if (data->image_start_offset + data->image_end_offset
> GST_BUFFER_SIZE (buffer)) {
@@ -104,10 +155,38 @@ mxf_up_handle_essence_element (const MXFUL * key, GstBuffer * buffer,
GST_BUFFER_DATA (buffer) += data->image_start_offset;
GST_BUFFER_SIZE (buffer) -= data->image_start_offset;
GST_BUFFER_SIZE (buffer) -= data->image_end_offset;
- *outbuf = buffer;
}
}
+ if (GST_BUFFER_SIZE (buffer) != data->bpp * data->width * data->height) {
+ GST_ERROR ("Invalid buffer size");
+ return GST_FLOW_ERROR;
+ }
+
+ if (data->bpp != 4
+ || GST_ROUND_UP_4 (data->width * data->bpp) != data->width * data->bpp) {
+ guint y;
+ GstBuffer *ret;
+ guint8 *indata, *outdata;
+
+ ret =
+ gst_buffer_new_and_alloc (GST_ROUND_UP_4 (data->width * data->bpp) *
+ data->height);
+ indata = GST_BUFFER_DATA (buffer);
+ outdata = GST_BUFFER_DATA (ret);
+
+ for (y = 0; y < data->height; y++) {
+ memcpy (outdata, indata, data->width * data->bpp);
+ outdata += GST_ROUND_UP_4 (data->width * data->bpp);
+ indata += data->width * data->bpp;
+ }
+
+ gst_buffer_unref (buffer);
+ *outbuf = ret;
+ } else {
+ *outbuf = buffer;
+ }
+
return GST_FLOW_OK;
}
@@ -117,85 +196,91 @@ mxf_up_rgba_create_caps (MXFMetadataTimelineTrack * track,
MXFEssenceElementHandleFunc * handler, gpointer * mapping_data)
{
GstCaps *caps = NULL;
+ guint i;
+ guint32 fourcc;
+ guint bpp;
if (!d->pixel_layout) {
GST_ERROR ("No pixel layout");
return NULL;
}
- if (d->n_pixel_layout == 3) {
- if (d->pixel_layout[0] == 'R' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'B' && d->pixel_layout[1] == 8
- && d->pixel_layout[3] == 8 && d->pixel_layout[5] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_RGB);
- } else if (d->pixel_layout[0] == 'B' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'R' && d->pixel_layout[1] == 8
- && d->pixel_layout[3] == 8 && d->pixel_layout[5] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_BGR);
- } else {
- GST_ERROR ("Unsupport 3 component pixel layout");
- return NULL;
- }
- } else if (d->n_pixel_layout == 4) {
- if (d->pixel_layout[0] == 'R' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'B' && d->pixel_layout[6] == 'F'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_RGBx);
- } else if (d->pixel_layout[0] == 'B' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'R' && d->pixel_layout[6] == 'F'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_BGRx);
- } else if (d->pixel_layout[0] == 'F' && d->pixel_layout[2] == 'R'
- && d->pixel_layout[4] == 'G' && d->pixel_layout[6] == 'B'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_xRGB);
- } else if (d->pixel_layout[0] == 'F' && d->pixel_layout[2] == 'B'
- && d->pixel_layout[4] == 'G' && d->pixel_layout[6] == 'R'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_xBGR);
- } else if (d->pixel_layout[0] == 'A' && d->pixel_layout[2] == 'R'
- && d->pixel_layout[4] == 'G' && d->pixel_layout[6] == 'B'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_ARGB);
- } else if (d->pixel_layout[0] == 'A' && d->pixel_layout[2] == 'B'
- && d->pixel_layout[4] == 'G' && d->pixel_layout[6] == 'R'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_ABGR);
- } else if (d->pixel_layout[0] == 'R' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'B' && d->pixel_layout[6] == 'A'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_RGBA);
- } else if (d->pixel_layout[0] == 'B' && d->pixel_layout[2] == 'G'
- && d->pixel_layout[4] == 'R' && d->pixel_layout[6] == 'A'
- && d->pixel_layout[1] == 8 && d->pixel_layout[3] == 8
- && d->pixel_layout[5] == 8 && d->pixel_layout[7] == 8) {
- caps = gst_caps_from_string (GST_VIDEO_CAPS_BGRA);
- } else {
- GST_ERROR ("Unsupport 4 component pixel layout");
- return NULL;
+ for (i = 0; i < G_N_ELEMENTS (_rgba_mapping_table); i++) {
+ if (d->n_pixel_layout != _rgba_mapping_table[i].n_pixel_layout)
+ continue;
+
+ if (memcmp (d->pixel_layout, &_rgba_mapping_table[i].pixel_layout,
+ _rgba_mapping_table[i].n_pixel_layout * 2) == 0) {
+ caps = gst_caps_from_string (_rgba_mapping_table[i].caps);
+ fourcc = _rgba_mapping_table[i].fourcc;
+ bpp = _rgba_mapping_table[i].n_pixel_layout;
+ break;
}
+ }
+
+ if (caps) {
+ MXFUPMappingData *data = g_new0 (MXFUPMappingData, 1);
+
+ mxf_metadata_generic_picture_essence_descriptor_set_caps (&d->parent, caps);
+
+ data->width = d->parent.stored_width;
+ data->height = d->parent.stored_height;
+ data->fourcc = fourcc;
+ data->bpp = bpp;
+ data->image_start_offset =
+ ((MXFMetadataGenericPictureEssenceDescriptor *) d)->image_start_offset;
+ data->image_end_offset =
+ ((MXFMetadataGenericPictureEssenceDescriptor *) d)->image_end_offset;
+
+ *mapping_data = data;
} else {
- GST_ERROR ("Pixel layouts with %u components not supported yet",
- d->n_pixel_layout);
- return NULL;
+ GST_WARNING ("Unsupported pixel layout");
+ }
+
+ return caps;
+}
+
+static GstCaps *
+mxf_up_cdci_create_caps (MXFMetadataTimelineTrack * track,
+ MXFMetadataCDCIPictureEssenceDescriptor * d, GstTagList ** tags,
+ MXFEssenceElementHandleFunc * handler, gpointer * mapping_data)
+{
+ GstCaps *caps = NULL;
+ guint i;
+ guint32 fourcc;
+ guint bpp;
+
+ for (i = 0; i < G_N_ELEMENTS (_cdci_mapping_table); i++) {
+ if (_cdci_mapping_table[i].horizontal_subsampling ==
+ d->horizontal_subsampling
+ && _cdci_mapping_table[i].vertical_subsampling ==
+ d->vertical_subsampling
+ && _cdci_mapping_table[i].reversed_byte_order ==
+ d->reversed_byte_order) {
+ caps = gst_caps_from_string (_cdci_mapping_table[i].caps);
+ fourcc = _cdci_mapping_table[i].fourcc;
+ bpp = _cdci_mapping_table[i].bpp;
+ break;
+ }
}
if (caps) {
MXFUPMappingData *data = g_new0 (MXFUPMappingData, 1);
+ mxf_metadata_generic_picture_essence_descriptor_set_caps (&d->parent, caps);
+
+ data->width = d->parent.stored_width;
+ data->height = d->parent.stored_height;
+ data->fourcc = fourcc;
+ data->bpp = bpp;
data->image_start_offset =
((MXFMetadataGenericPictureEssenceDescriptor *) d)->image_start_offset;
data->image_end_offset =
((MXFMetadataGenericPictureEssenceDescriptor *) d)->image_end_offset;
*mapping_data = data;
+ } else {
+ GST_WARNING ("Unsupported CDCI format");
}
return caps;
@@ -247,15 +332,12 @@ mxf_up_create_caps (MXFMetadataTimelineTrack * track, GstTagList ** tags,
if (r) {
caps = mxf_up_rgba_create_caps (track, r, tags, handler, mapping_data);
+ } else if (c) {
+ caps = mxf_up_cdci_create_caps (track, c, tags, handler, mapping_data);
} else {
- GST_ERROR ("CDCI uncompressed picture essence not supported yet");
return NULL;
}
- if (caps) {
- mxf_metadata_generic_picture_essence_descriptor_set_caps (p, caps);
- }
-
return caps;
}
@@ -264,8 +346,241 @@ static const MXFEssenceElementHandler mxf_up_essence_element_handler = {
mxf_up_create_caps
};
+static GstFlowReturn
+mxf_up_write_func (GstBuffer * buffer, GstCaps * caps, gpointer mapping_data,
+ GstAdapter * adapter, GstBuffer ** outbuf, gboolean flush)
+{
+ MXFUPMappingData *data = mapping_data;
+
+ if (!buffer)
+ return GST_FLOW_OK;
+
+ if (GST_BUFFER_SIZE (buffer) !=
+ GST_ROUND_UP_4 (data->bpp * data->width) * data->height) {
+ GST_ERROR ("Invalid buffer size");
+ return GST_FLOW_ERROR;
+ }
+
+ if (data->bpp != 4
+ || GST_ROUND_UP_4 (data->width * data->bpp) != data->width * data->bpp) {
+ guint y;
+ GstBuffer *ret;
+ guint8 *indata, *outdata;
+
+ ret = gst_buffer_new_and_alloc (data->width * data->bpp * data->height);
+ indata = GST_BUFFER_DATA (buffer);
+ outdata = GST_BUFFER_DATA (ret);
+
+ for (y = 0; y < data->height; y++) {
+ memcpy (outdata, indata, data->width * data->bpp);
+ indata += GST_ROUND_UP_4 (data->width * data->bpp);
+ outdata += data->width * data->bpp;
+ }
+
+ gst_buffer_unref (buffer);
+
+ *outbuf = ret;
+ } else {
+ *outbuf = buffer;
+ }
+
+ return GST_FLOW_OK;
+}
+
+static const guint8 up_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x01,
+ 0x0D, 0x01, 0x03, 0x01, 0x02, 0x05, 0x7F, 0x01
+};
+
+static MXFMetadataFileDescriptor *
+mxf_up_get_rgba_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataRGBAPictureEssenceDescriptor *ret;
+ guint i;
+ GstCaps *tmp, *intersection;
+ MXFUPMappingData *md = g_new0 (MXFUPMappingData, 1);
+
+ *mapping_data = md;
+
+ ret = (MXFMetadataRGBAPictureEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_RGBA_PICTURE_ESSENCE_DESCRIPTOR);
+
+ for (i = 0; i < G_N_ELEMENTS (_rgba_mapping_table); i++) {
+ tmp = gst_caps_from_string (_rgba_mapping_table[i].caps);
+ intersection = gst_caps_intersect (caps, tmp);
+ gst_caps_unref (tmp);
+
+ if (!gst_caps_is_empty (intersection)) {
+ gst_caps_unref (intersection);
+ ret->n_pixel_layout = _rgba_mapping_table[i].n_pixel_layout;
+ ret->pixel_layout = g_new0 (guint8, ret->n_pixel_layout * 2);
+ md->fourcc = _rgba_mapping_table[i].fourcc;
+ md->bpp = _rgba_mapping_table[i].n_pixel_layout;
+ memcpy (ret->pixel_layout, _rgba_mapping_table[i].pixel_layout,
+ ret->n_pixel_layout * 2);
+ break;
+ }
+ gst_caps_unref (intersection);
+ }
+
+ if (md->fourcc == 0) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+
+ memcpy (&ret->parent.parent.essence_container, &up_essence_container_ul, 16);
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ md->width = ret->parent.stored_width;
+ md->height = ret->parent.stored_height;
+
+ *handler = mxf_up_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static MXFMetadataFileDescriptor *
+mxf_up_get_cdci_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataCDCIPictureEssenceDescriptor *ret;
+ guint i;
+ GstCaps *tmp, *intersection;
+ MXFUPMappingData *md = g_new0 (MXFUPMappingData, 1);
+
+ *mapping_data = md;
+
+ ret = (MXFMetadataCDCIPictureEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_CDCI_PICTURE_ESSENCE_DESCRIPTOR);
+
+ for (i = 0; i < G_N_ELEMENTS (_cdci_mapping_table); i++) {
+ tmp = gst_caps_from_string (_cdci_mapping_table[i].caps);
+ intersection = gst_caps_intersect (caps, tmp);
+ gst_caps_unref (tmp);
+
+ if (!gst_caps_is_empty (intersection)) {
+ gst_caps_unref (intersection);
+ ret->horizontal_subsampling =
+ _cdci_mapping_table[i].horizontal_subsampling;
+ ret->vertical_subsampling = _cdci_mapping_table[i].vertical_subsampling;
+ ret->reversed_byte_order = _cdci_mapping_table[i].reversed_byte_order;
+ md->fourcc = _cdci_mapping_table[i].fourcc;
+ md->bpp = _cdci_mapping_table[i].bpp;
+ break;
+ }
+ gst_caps_unref (intersection);
+ }
+
+ if (md->fourcc == 0) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ memcpy (&ret->parent.parent.essence_container, &up_essence_container_ul, 16);
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ md->width = ret->parent.stored_width;
+ md->height = ret->parent.stored_height;
+
+ *handler = mxf_up_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static MXFMetadataFileDescriptor *
+mxf_up_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ GstStructure *s;
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "video/x-raw-rgb") == 0) {
+ return mxf_up_get_rgba_descriptor (tmpl, caps, handler, mapping_data);
+ } else if (strcmp (gst_structure_get_name (s), "video/x-raw-yuv") == 0) {
+ guint32 fourcc;
+
+ if (!gst_structure_get_fourcc (s, "format", &fourcc))
+ return NULL;
+
+ if (fourcc == GST_MAKE_FOURCC ('A', 'Y', 'U', 'V') ||
+ fourcc == GST_MAKE_FOURCC ('v', '3', '0', '8'))
+ return mxf_up_get_rgba_descriptor (tmpl, caps, handler, mapping_data);
+
+ return mxf_up_get_cdci_descriptor (tmpl, caps, handler, mapping_data);
+ }
+
+ g_assert_not_reached ();
+}
+
+static void
+mxf_up_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_up_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ edit_rate->n = a->sample_rate.n;
+ edit_rate->d = a->sample_rate.d;
+}
+
+static guint32
+mxf_up_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x15 << 24) | (0x02 << 8);
+}
+
+static MXFEssenceElementWriter mxf_up_essence_element_writer = {
+ mxf_up_get_descriptor,
+ mxf_up_update_descriptor,
+ mxf_up_get_edit_rate,
+ mxf_up_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
void
mxf_up_init (void)
{
mxf_essence_element_handler_register (&mxf_up_essence_element_handler);
+
+ mxf_up_essence_element_writer.pad_template =
+ gst_pad_template_new ("up_video_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string (GST_VIDEO_CAPS_RGB "; "
+ GST_VIDEO_CAPS_BGR "; "
+ GST_VIDEO_CAPS_RGBx "; "
+ GST_VIDEO_CAPS_xRGB "; "
+ GST_VIDEO_CAPS_BGRx "; "
+ GST_VIDEO_CAPS_xBGR "; "
+ GST_VIDEO_CAPS_ARGB "; "
+ GST_VIDEO_CAPS_RGBA "; "
+ GST_VIDEO_CAPS_ABGR "; "
+ GST_VIDEO_CAPS_BGRA "; "
+ GST_VIDEO_CAPS_YUV ("AYUV") "; "
+ GST_VIDEO_CAPS_YUV ("v308") "; "
+ GST_VIDEO_CAPS_YUV ("UYVY") "; " GST_VIDEO_CAPS_YUV ("YUY2")));
+
+ memcpy (&mxf_up_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_PICTURE_ESSENCE),
+ 16);
+ mxf_essence_element_writer_register (&mxf_up_essence_element_writer);
}
diff --git a/gst/mxf/mxfvc3.c b/gst/mxf/mxfvc3.c
index e0f9b804..3e94f378 100644
--- a/gst/mxf/mxfvc3.c
+++ b/gst/mxf/mxfvc3.c
@@ -26,9 +26,11 @@
#endif
#include <gst/gst.h>
+#include <gst/video/video.h>
#include <string.h>
#include "mxfvc3.h"
+#include "mxfwrite.h"
GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
#define GST_CAT_DEFAULT mxf_debug
@@ -154,8 +156,93 @@ static const MXFEssenceElementHandler mxf_vc3_essence_element_handler = {
mxf_vc3_create_caps
};
+static GstFlowReturn
+mxf_vc3_write_func (GstBuffer * buffer, GstCaps * caps, gpointer mapping_data,
+ GstAdapter * adapter, GstBuffer ** outbuf, gboolean flush)
+{
+ *outbuf = buffer;
+ return GST_FLOW_OK;
+}
+
+/* FIXME: In which version was this added? Byte 7, assuming version 10 */
+static const guint8 vc3_essence_container_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x0A,
+ 0x0d, 0x01, 0x03, 0x01, 0x02, 0x11, 0x01, 0x00
+};
+
+static MXFMetadataFileDescriptor *
+mxf_vc3_get_descriptor (GstPadTemplate * tmpl, GstCaps * caps,
+ MXFEssenceElementWriteFunc * handler, gpointer * mapping_data)
+{
+ MXFMetadataCDCIPictureEssenceDescriptor *ret;
+ GstStructure *s;
+
+ s = gst_caps_get_structure (caps, 0);
+ if (strcmp (gst_structure_get_name (s), "video/x-dnxhd") != 0) {
+ GST_ERROR ("Invalid caps %" GST_PTR_FORMAT, caps);
+ return NULL;
+ }
+
+ ret = (MXFMetadataCDCIPictureEssenceDescriptor *)
+ gst_mini_object_new (MXF_TYPE_METADATA_CDCI_PICTURE_ESSENCE_DESCRIPTOR);
+
+ memcpy (&ret->parent.parent.essence_container, &vc3_essence_container_ul, 16);
+
+ if (!mxf_metadata_generic_picture_essence_descriptor_from_caps (&ret->parent,
+ caps)) {
+ gst_mini_object_unref (GST_MINI_OBJECT_CAST (ret));
+ return NULL;
+ }
+
+ *handler = mxf_vc3_write_func;
+
+ return (MXFMetadataFileDescriptor *) ret;
+}
+
+static void
+mxf_vc3_update_descriptor (MXFMetadataFileDescriptor * d, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf)
+{
+ return;
+}
+
+static void
+mxf_vc3_get_edit_rate (MXFMetadataFileDescriptor * a, GstCaps * caps,
+ gpointer mapping_data, GstBuffer * buf, MXFMetadataSourcePackage * package,
+ MXFMetadataTimelineTrack * track, MXFFraction * edit_rate)
+{
+ edit_rate->n = a->sample_rate.n;
+ edit_rate->d = a->sample_rate.d;
+}
+
+static guint32
+mxf_vc3_get_track_number_template (MXFMetadataFileDescriptor * a,
+ GstCaps * caps, gpointer mapping_data)
+{
+ return (0x15 << 24) | (0x05 << 8);
+}
+
+static MXFEssenceElementWriter mxf_vc3_essence_element_writer = {
+ mxf_vc3_get_descriptor,
+ mxf_vc3_update_descriptor,
+ mxf_vc3_get_edit_rate,
+ mxf_vc3_get_track_number_template,
+ NULL,
+ {{0,}}
+};
+
void
mxf_vc3_init (void)
{
mxf_essence_element_handler_register (&mxf_vc3_essence_element_handler);
+
+ mxf_vc3_essence_element_writer.pad_template =
+ gst_pad_template_new ("vc3_video_sink_%u", GST_PAD_SINK, GST_PAD_REQUEST,
+ gst_caps_from_string ("video/x-dnxhd, width = " GST_VIDEO_SIZE_RANGE
+ ", height = " GST_VIDEO_SIZE_RANGE ", framerate = "
+ GST_VIDEO_FPS_RANGE));
+ memcpy (&mxf_vc3_essence_element_writer.data_definition,
+ mxf_metadata_track_identifier_get (MXF_METADATA_TRACK_PICTURE_ESSENCE),
+ 16);
+ mxf_essence_element_writer_register (&mxf_vc3_essence_element_writer);
}
diff --git a/gst/mxf/mxfwrite.c b/gst/mxf/mxfwrite.c
new file mode 100644
index 00000000..c2c15bba
--- /dev/null
+++ b/gst/mxf/mxfwrite.c
@@ -0,0 +1,567 @@
+/* GStreamer
+ * Copyright (C) 2009 Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <gst/gst.h>
+#include <string.h>
+
+#include "mxfwrite.h"
+
+GST_DEBUG_CATEGORY_EXTERN (mxf_debug);
+#define GST_CAT_DEFAULT mxf_debug
+
+static GList *_essence_element_writer_registry = NULL;
+static GPtrArray *_essence_element_writer_pad_templates = NULL;
+
+void
+mxf_essence_element_writer_register (const MXFEssenceElementWriter * writer)
+{
+ _essence_element_writer_registry =
+ g_list_prepend (_essence_element_writer_registry, (gpointer) writer);
+
+ if (!_essence_element_writer_pad_templates)
+ _essence_element_writer_pad_templates = g_ptr_array_new ();
+
+ if (_essence_element_writer_pad_templates->len > 0 &&
+ g_ptr_array_index (_essence_element_writer_pad_templates,
+ _essence_element_writer_pad_templates->len - 1) == NULL)
+ g_ptr_array_remove_index (_essence_element_writer_pad_templates,
+ _essence_element_writer_pad_templates->len - 1);
+
+ g_ptr_array_add (_essence_element_writer_pad_templates,
+ (gpointer) writer->pad_template);
+}
+
+const GstPadTemplate **
+mxf_essence_element_writer_get_pad_templates (void)
+{
+ if (!_essence_element_writer_pad_templates
+ || _essence_element_writer_pad_templates->len == 0)
+ return NULL;
+
+ if (g_ptr_array_index (_essence_element_writer_pad_templates,
+ _essence_element_writer_pad_templates->len - 1))
+ g_ptr_array_add (_essence_element_writer_pad_templates, NULL);
+
+ return (const GstPadTemplate **) _essence_element_writer_pad_templates->pdata;
+}
+
+const MXFEssenceElementWriter *
+mxf_essence_element_writer_find (const GstPadTemplate * templ)
+{
+ GList *l = _essence_element_writer_registry;
+
+ for (; l; l = l->next) {
+ MXFEssenceElementWriter *writer = l->data;
+
+ if (writer->pad_template == templ)
+ return writer;
+ }
+
+ return NULL;
+}
+
+void
+mxf_ul_set (MXFUL * ul, GHashTable * hashtable)
+{
+ guint i;
+
+next_try:
+ for (i = 0; i < 4; i++)
+ GST_WRITE_UINT32_BE (&ul->u[i * 4], g_random_int ());
+
+ if (hashtable && g_hash_table_lookup_extended (hashtable, ul, NULL, NULL))
+ goto next_try;
+}
+
+void
+mxf_umid_set (MXFUMID * umid)
+{
+ guint i;
+ guint32 tmp;
+
+ /* SMPTE S330M 5.1.1:
+ * UMID Identifier
+ */
+ umid->u[0] = 0x06;
+ umid->u[1] = 0x0a;
+ umid->u[2] = 0x2b;
+ umid->u[3] = 0x34;
+ umid->u[4] = 0x01;
+ umid->u[5] = 0x01;
+ umid->u[6] = 0x01;
+ umid->u[7] = 0x05; /* version, see RP210 */
+ umid->u[8] = 0x01;
+ umid->u[9] = 0x01;
+ umid->u[10] = 0x0d; /* mixed group of components in a single container */
+
+ /* - UUID/UL method for material number
+ * - 24 bit PRG for instance number
+ */
+ umid->u[11] = 0x20 | 0x02;
+
+ /* Length of remaining data */
+ umid->u[12] = 0x13;
+
+ /* Instance number */
+ tmp = g_random_int ();
+ umid->u[13] = (tmp >> 24) & 0xff;
+ umid->u[14] = (tmp >> 16) & 0xff;
+ umid->u[15] = (tmp >> 8) & 0xff;
+
+ /* Material number: ISO UUID Version 4 */
+ for (i = 16; i < 32; i += 4)
+ GST_WRITE_UINT32_BE (&umid->u[i], g_random_int ());
+
+ umid->u[16 + 6] &= 0x0f;
+ umid->u[16 + 6] |= 0x40;
+
+ umid->u[16 + 8] &= 0x3f;
+ umid->u[16 + 8] |= 0x80;
+}
+
+void
+mxf_timestamp_set_now (MXFTimestamp * timestamp)
+{
+ GTimeVal tv;
+ time_t t;
+ struct tm *tm;
+
+#ifdef HAVE_GMTIME_R
+ struct tm tm_;
+#endif
+
+ g_get_current_time (&tv);
+ t = (time_t) tv.tv_sec;
+
+#ifdef HAVE_GMTIME_R
+ tm = gmtime_r (&t, &tm_);
+#else
+ tm = gmtime (&t);
+#endif
+
+ timestamp->year = tm->tm_year + 1900;
+ timestamp->month = tm->tm_mon;
+ timestamp->day = tm->tm_mday;
+ timestamp->hour = tm->tm_hour;
+ timestamp->minute = tm->tm_min;
+ timestamp->second = tm->tm_sec;
+ timestamp->msecond = tv.tv_usec / 1000;
+}
+
+static guint8 mxf_op_identification_ul[] = {
+ 0x06, 0x0e, 0x2b, 0x34, 0x04, 0x01, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01
+};
+
+void
+mxf_op_set_atom (MXFUL * ul, gboolean single_sourceclip,
+ gboolean single_essence_track)
+{
+ memcpy (&ul->u, &mxf_op_identification_ul, 12);
+ ul->u[12] = 0x10;
+ ul->u[13] = 0;
+
+ if (!single_sourceclip)
+ ul->u[13] |= 0x80;
+
+ if (!single_essence_track)
+ ul->u[13] |= 0x40;
+
+ ul->u[14] = 0;
+ ul->u[15] = 0;
+}
+
+void
+mxf_op_set_generalized (MXFUL * ul, MXFOperationalPattern pattern,
+ gboolean internal_essence, gboolean streamable, gboolean single_track)
+{
+ g_return_if_fail (pattern >= MXF_OP_1a);
+
+ memcpy (&ul->u, &mxf_op_identification_ul, 12);
+
+ if (pattern == MXF_OP_1a || pattern == MXF_OP_1b || pattern == MXF_OP_1c)
+ ul->u[12] = 0x01;
+ else if (pattern == MXF_OP_2a || pattern == MXF_OP_2b || pattern == MXF_OP_2c)
+ ul->u[12] = 0x02;
+ else if (pattern == MXF_OP_3a || pattern == MXF_OP_3b || pattern == MXF_OP_3c)
+ ul->u[12] = 0x03;
+
+ if (pattern == MXF_OP_1a || pattern == MXF_OP_2a || pattern == MXF_OP_3a)
+ ul->u[13] = 0x01;
+ else if (pattern == MXF_OP_1b || pattern == MXF_OP_2b || pattern == MXF_OP_3b)
+ ul->u[13] = 0x02;
+ else if (pattern == MXF_OP_1c || pattern == MXF_OP_2c || pattern == MXF_OP_3c)
+ ul->u[13] = 0x02;
+
+ ul->u[14] = 0x80;
+ if (!internal_essence)
+ ul->u[14] |= 0x40;
+ if (!streamable)
+ ul->u[14] |= 0x20;
+ if (!single_track)
+ ul->u[14] |= 0x10;
+
+ ul->u[15] = 0;
+}
+
+static void
+_mxf_mapping_ul_free (MXFUL * ul)
+{
+ g_slice_free (MXFUL, ul);
+}
+
+guint16
+mxf_primer_pack_add_mapping (MXFPrimerPack * primer, guint16 local_tag,
+ const MXFUL * ul)
+{
+ MXFUL *uid;
+#ifndef GST_DISABLE_GST_DEBUG
+ gchar str[48];
+#endif
+
+ if (primer->mappings == NULL) {
+ primer->mappings = g_hash_table_new_full (g_direct_hash, g_direct_equal,
+ (GDestroyNotify) NULL, (GDestroyNotify) _mxf_mapping_ul_free);
+ }
+
+ if (primer->reverse_mappings == NULL) {
+ primer->reverse_mappings = g_hash_table_new_full ((GHashFunc) mxf_ul_hash,
+ (GEqualFunc) mxf_ul_is_equal, (GDestroyNotify) _mxf_mapping_ul_free,
+ (GDestroyNotify) NULL);
+ }
+
+ if (primer->next_free_tag == 0xffff && local_tag == 0) {
+ GST_ERROR ("Used too many dynamic tags");
+ return 0;
+ }
+
+ if (local_tag == 0) {
+ guint16 tmp;
+
+ tmp = GPOINTER_TO_UINT (g_hash_table_lookup (primer->reverse_mappings, ul));
+ if (tmp == 0) {
+ local_tag = primer->next_free_tag;
+ primer->next_free_tag++;
+ }
+ } else {
+ if (g_hash_table_lookup (primer->mappings, GUINT_TO_POINTER (local_tag)))
+ return local_tag;
+ }
+
+ g_assert (local_tag != 0);
+
+ uid = g_slice_new (MXFUL);
+ memcpy (uid, ul, 16);
+
+ GST_DEBUG ("Adding mapping = 0x%04x -> %s", local_tag,
+ mxf_ul_to_string (uid, str));
+ g_hash_table_insert (primer->mappings, GUINT_TO_POINTER (local_tag), uid);
+ uid = g_slice_dup (MXFUL, uid);
+ g_hash_table_insert (primer->reverse_mappings, uid,
+ GUINT_TO_POINTER (local_tag));
+
+ return local_tag;
+}
+
+guint
+mxf_ber_encode_size (guint size, guint8 ber[9])
+{
+ guint8 slen, i;
+ guint8 tmp[8];
+
+ memset (ber, 0, 9);
+
+ if (size <= 127) {
+ ber[0] = size;
+ return 1;
+ } else if (size > G_MAXUINT) {
+ return 0;
+ }
+
+ slen = 0;
+ while (size > 0) {
+ tmp[slen] = size & 0xff;
+ size >>= 8;
+ slen++;
+ }
+
+ ber[0] = 0x80 | slen;
+ for (i = 0; i < slen; i++) {
+ ber[i + 1] = tmp[slen - i - 1];
+ }
+
+ return slen + 1;
+}
+
+void
+mxf_timestamp_write (const MXFTimestamp * timestamp, guint8 * data)
+{
+ GST_WRITE_UINT16_BE (data, timestamp->year);
+ GST_WRITE_UINT8 (data + 2, timestamp->month);
+ GST_WRITE_UINT8 (data + 3, timestamp->day);
+ GST_WRITE_UINT8 (data + 4, timestamp->hour);
+ GST_WRITE_UINT8 (data + 5, timestamp->minute);
+ GST_WRITE_UINT8 (data + 6, timestamp->second);
+ GST_WRITE_UINT8 (data + 7, (timestamp->msecond * 256) / 1000);
+}
+
+guint8 *
+mxf_utf8_to_utf16 (const gchar * str, guint16 * size)
+{
+ guint8 *ret;
+ GError *error = NULL;
+ gsize s;
+
+ g_return_val_if_fail (size != NULL, NULL);
+
+ if (str == NULL) {
+ *size = 0;
+ return NULL;
+ }
+
+ ret = (guint8 *)
+ g_convert_with_fallback (str, -1, "UTF-16BE", "UTF-8", "*", NULL, &s,
+ &error);
+
+ if (ret == NULL) {
+ GST_WARNING ("UTF-16-BE to UTF-8 conversion failed: %s", error->message);
+ g_error_free (error);
+ *size = 0;
+ return NULL;
+ }
+
+ *size = s;
+ return (guint8 *) ret;
+}
+
+void
+mxf_product_version_write (const MXFProductVersion * version, guint8 * data)
+{
+ GST_WRITE_UINT16_BE (data, version->major);
+ GST_WRITE_UINT16_BE (data + 2, version->minor);
+ GST_WRITE_UINT16_BE (data + 4, version->patch);
+ GST_WRITE_UINT16_BE (data + 6, version->build);
+ GST_WRITE_UINT16_BE (data + 8, version->release);
+}
+
+GstBuffer *
+mxf_partition_pack_to_buffer (const MXFPartitionPack * pack)
+{
+ static const guint8 partition_pack_ul[] =
+ { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01
+ };
+ guint slen;
+ guint8 ber[9];
+ GstBuffer *ret;
+ guint8 *data;
+ guint i;
+ guint size =
+ 8 + 16 * pack->n_essence_containers + 16 + 4 + 8 + 4 + 8 + 8 + 8 + 8 + 8 +
+ 4 + 2 + 2;
+
+ slen = mxf_ber_encode_size (size, ber);
+
+ ret = gst_buffer_new_and_alloc (16 + slen + size);
+ memcpy (GST_BUFFER_DATA (ret), &partition_pack_ul, 13);
+ if (pack->type == MXF_PARTITION_PACK_HEADER)
+ GST_BUFFER_DATA (ret)[13] = 0x02;
+ else if (pack->type == MXF_PARTITION_PACK_BODY)
+ GST_BUFFER_DATA (ret)[13] = 0x03;
+ else if (pack->type == MXF_PARTITION_PACK_FOOTER)
+ GST_BUFFER_DATA (ret)[13] = 0x04;
+ GST_BUFFER_DATA (ret)[14] = 0;
+ if (pack->complete)
+ GST_BUFFER_DATA (ret)[14] |= 0x02;
+ if (pack->closed)
+ GST_BUFFER_DATA (ret)[14] |= 0x01;
+ GST_BUFFER_DATA (ret)[14] += 1;
+ GST_BUFFER_DATA (ret)[15] = 0;
+ memcpy (GST_BUFFER_DATA (ret) + 16, &ber, slen);
+
+ data = GST_BUFFER_DATA (ret) + 16 + slen;
+
+ GST_WRITE_UINT16_BE (data, pack->major_version);
+ GST_WRITE_UINT16_BE (data + 2, pack->minor_version);
+ data += 4;
+
+ GST_WRITE_UINT32_BE (data, pack->kag_size);
+ data += 4;
+
+ GST_WRITE_UINT64_BE (data, pack->this_partition);
+ data += 8;
+
+ GST_WRITE_UINT64_BE (data, pack->prev_partition);
+ data += 8;
+
+ GST_WRITE_UINT64_BE (data, pack->footer_partition);
+ data += 8;
+
+ GST_WRITE_UINT64_BE (data, pack->header_byte_count);
+ data += 8;
+
+ GST_WRITE_UINT64_BE (data, pack->index_byte_count);
+ data += 8;
+
+ GST_WRITE_UINT32_BE (data, pack->index_sid);
+ data += 4;
+
+ GST_WRITE_UINT64_BE (data, pack->body_offset);
+ data += 8;
+
+ GST_WRITE_UINT32_BE (data, pack->body_sid);
+ data += 4;
+
+ memcpy (data, &pack->operational_pattern, 16);
+ data += 16;
+
+ GST_WRITE_UINT32_BE (data, pack->n_essence_containers);
+ GST_WRITE_UINT32_BE (data + 4, 16);
+ data += 8;
+
+ for (i = 0; i < pack->n_essence_containers; i++)
+ memcpy (data + 16 * i, &pack->essence_containers[i], 16);
+
+ return ret;
+}
+
+GstBuffer *
+mxf_primer_pack_to_buffer (const MXFPrimerPack * pack)
+{
+ static const guint8 primer_pack_ul[] =
+ { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01, 0x0d, 0x01, 0x02, 0x01,
+ 0x01, 0x05, 0x01, 0x00
+ };
+ guint slen;
+ guint8 ber[9];
+ GstBuffer *ret;
+ guint n;
+ guint8 *data;
+
+ if (pack->mappings)
+ n = g_hash_table_size (pack->mappings);
+ else
+ n = 0;
+
+ slen = mxf_ber_encode_size (8 + 18 * n, ber);
+
+ ret = gst_buffer_new_and_alloc (16 + slen + 8 + 18 * n);
+ memcpy (GST_BUFFER_DATA (ret), &primer_pack_ul, 16);
+ memcpy (GST_BUFFER_DATA (ret) + 16, &ber, slen);
+
+ data = GST_BUFFER_DATA (ret) + 16 + slen;
+
+ GST_WRITE_UINT32_BE (data, n);
+ GST_WRITE_UINT32_BE (data + 4, 18);
+ data += 8;
+
+ if (pack->mappings) {
+ guint16 local_tag;
+ MXFUL *ul;
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ GHashTableIter iter;
+
+ g_hash_table_iter_init (&iter, pack->mappings);
+#else
+ GList *l, *values;
+
+ keys = g_hash_table_get_keys (pack->mappings);
+#endif
+
+#if GLIB_CHECK_VERSION (2, 16, 0)
+ while (g_hash_table_iter_next (&iter, (gpointer) & local_tag,
+ (gpointer) & ul)) {
+#else
+ for (l = keys l; l = l->next) {
+ local_tag = GPOINTER_TO_GUINT (l->data);
+ ul = g_hash_table_lookup (pack->mappings, GUINT_TO_POINTER (local_tag));
+#endif
+ GST_WRITE_UINT16_BE (data, local_tag);
+ memcpy (data + 2, ul, 16);
+ data += 18;
+ }
+
+#if !GLIB_CHECK_VERSION (2, 16, 0)
+ g_list_free (keys);
+#endif
+ }
+
+ return ret;
+}
+
+GstBuffer *
+mxf_fill_new (guint size)
+{
+ static const guint8 fill_ul[] =
+ { 0x06, 0x0e, 0x2b, 0x34, 0x01, 0x01, 0x01, 0x01,
+ 0x03, 0x01, 0x02, 0x10, 0x01, 0x00, 0x00, 0x00
+ };
+ GstBuffer *ret;
+ guint slen;
+ guint8 ber[9];
+
+ slen = mxf_ber_encode_size (size, ber);
+
+ ret = gst_buffer_new_and_alloc (16 + slen + size);
+ memcpy (GST_BUFFER_DATA (ret), &fill_ul, 16);
+ memcpy (GST_BUFFER_DATA (ret) + 16, &ber, slen);
+ memset (GST_BUFFER_DATA (ret) + slen, 0, size);
+
+ return ret;
+}
+
+static const guint8 random_index_pack_ul[] =
+ { 0x06, 0x0e, 0x2b, 0x34, 0x02, 0x05, 0x01, 0x01,
+ 0x0d, 0x01, 0x02, 0x01, 0x01, 0x11, 0x01, 0x00
+};
+
+GstBuffer *
+mxf_random_index_pack_to_buffer (const GArray * array)
+{
+ MXFRandomIndexPackEntry *entry;
+ guint i;
+ GstBuffer *ret;
+ guint8 slen, ber[9];
+ guint size;
+ guint8 *data;
+
+ if (array->len == 0)
+ return NULL;
+
+ size = array->len * 12 + 4;
+ slen = mxf_ber_encode_size (size, ber);
+ ret = gst_buffer_new_and_alloc (16 + slen + size);
+ memcpy (GST_BUFFER_DATA (ret), random_index_pack_ul, 16);
+ memcpy (GST_BUFFER_DATA (ret) + 16, ber, slen);
+
+ data = GST_BUFFER_DATA (ret) + 16 + slen;
+
+ for (i = 0; i < array->len; i++) {
+ entry = &g_array_index (array, MXFRandomIndexPackEntry, i);
+ GST_WRITE_UINT32_BE (data, entry->body_sid);
+ GST_WRITE_UINT64_BE (data + 4, entry->offset);
+ data += 12;
+ }
+ GST_WRITE_UINT32_BE (data, GST_BUFFER_SIZE (ret));
+
+ return ret;
+}
diff --git a/gst/mxf/mxfwrite.h b/gst/mxf/mxfwrite.h
new file mode 100644
index 00000000..4ebca925
--- /dev/null
+++ b/gst/mxf/mxfwrite.h
@@ -0,0 +1,85 @@
+/* GStreamer
+ * Copyright (C) 2009 Sebastian Dröge <sebastian.droege@collabora.co.uk>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+/* Handling of the basic MXF types */
+
+#ifndef __MXF_WRITE_H__
+#define __MXF_WRITE_H__
+
+#include <string.h>
+#include <gst/gst.h>
+#include <gst/base/gstadapter.h>
+
+#include "mxfmetadata.h"
+#include "mxftypes.h"
+#include "mxfparse.h"
+
+typedef GstFlowReturn (*MXFEssenceElementWriteFunc) (GstBuffer *buffer, GstCaps *caps, gpointer mapping_data, GstAdapter *adapter, GstBuffer **outbuf, gboolean flush);
+
+typedef struct {
+ MXFMetadataFileDescriptor * (*get_descriptor) (GstPadTemplate *tmpl, GstCaps *caps, MXFEssenceElementWriteFunc *handler, gpointer *mapping_data);
+ void (*update_descriptor) (MXFMetadataFileDescriptor *d, GstCaps *caps, gpointer mapping_data, GstBuffer *buf);
+ void (*get_edit_rate) (MXFMetadataFileDescriptor *a, GstCaps *caps, gpointer mapping_data, GstBuffer *buf, MXFMetadataSourcePackage *package, MXFMetadataTimelineTrack *track, MXFFraction *edit_rate);
+ guint32 (*get_track_number_template) (MXFMetadataFileDescriptor *a, GstCaps *caps, gpointer mapping_data);
+ const GstPadTemplate *pad_template;
+ MXFUL data_definition;
+} MXFEssenceElementWriter;
+
+typedef enum {
+ MXF_OP_UNKNOWN = 0,
+ MXF_OP_ATOM,
+ MXF_OP_1a,
+ MXF_OP_1b,
+ MXF_OP_1c,
+ MXF_OP_2a,
+ MXF_OP_2b,
+ MXF_OP_2c,
+ MXF_OP_3a,
+ MXF_OP_3b,
+ MXF_OP_3c,
+} MXFOperationalPattern;
+
+void mxf_essence_element_writer_register (const MXFEssenceElementWriter *writer);
+const GstPadTemplate ** mxf_essence_element_writer_get_pad_templates (void);
+const MXFEssenceElementWriter *mxf_essence_element_writer_find (const GstPadTemplate *templ);
+
+void mxf_ul_set (MXFUL *ul, GHashTable *hashtable);
+void mxf_umid_set (MXFUMID *umid);
+
+void mxf_timestamp_set_now (MXFTimestamp *timestamp);
+void mxf_timestamp_write (const MXFTimestamp *timestamp, guint8 *data);
+
+void mxf_op_set_atom (MXFUL *ul, gboolean single_sourceclip, gboolean single_essence_track);
+void mxf_op_set_generalized (MXFUL *ul, MXFOperationalPattern pattern, gboolean internal_essence, gboolean streamable, gboolean single_track);
+
+guint16 mxf_primer_pack_add_mapping (MXFPrimerPack *primer, guint16 local_tag, const MXFUL *ul);
+
+guint mxf_ber_encode_size (guint size, guint8 ber[9]);
+
+guint8 * mxf_utf8_to_utf16 (const gchar *str, guint16 *size);
+
+void mxf_product_version_write (const MXFProductVersion *version, guint8 *data);
+
+GstBuffer * mxf_partition_pack_to_buffer (const MXFPartitionPack *pack);
+GstBuffer * mxf_primer_pack_to_buffer (const MXFPrimerPack *pack);
+GstBuffer * mxf_fill_new (guint size);
+
+GstBuffer * mxf_random_index_pack_to_buffer (const GArray *array);
+
+#endif /* __MXF_WRITE_H__ */