mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-18 13:25:56 +00:00
gst/qtdemux/qtdemux.c: Don't make rounding errors in timestamp/duration calculations.
Original commit message from CVS: * gst/qtdemux/qtdemux.c: (gst_qtdemux_prepare_current_sample), (gst_qtdemux_chain), (gst_qtdemux_add_stream), (qtdemux_dump_stsz), (qtdemux_dump_stco), (qtdemux_parse_trak): Don't make rounding errors in timestamp/duration calculations. Fix timestamps for AMR and IMA4. Fixes (#337436). Create a dummy segment even when there is no edit list.
This commit is contained in:
parent
bd72d30ce9
commit
e843de1d06
2 changed files with 153 additions and 133 deletions
|
@ -1,3 +1,12 @@
|
||||||
|
2006-04-08 Wim Taymans <wim@fluendo.com>
|
||||||
|
|
||||||
|
* gst/qtdemux/qtdemux.c: (gst_qtdemux_prepare_current_sample),
|
||||||
|
(gst_qtdemux_chain), (gst_qtdemux_add_stream), (qtdemux_dump_stsz),
|
||||||
|
(qtdemux_dump_stco), (qtdemux_parse_trak):
|
||||||
|
Don't make rounding errors in timestamp/duration calculations.
|
||||||
|
Fix timestamps for AMR and IMA4. Fixes (#337436).
|
||||||
|
Create a dummy segment even when there is no edit list.
|
||||||
|
|
||||||
2006-04-07 Julien MOUTTE <julien@moutte.net>
|
2006-04-07 Julien MOUTTE <julien@moutte.net>
|
||||||
|
|
||||||
* docs/plugins/gst-plugins-bad-plugins-decl-list.txt: Updates.
|
* docs/plugins/gst-plugins-bad-plugins-decl-list.txt: Updates.
|
||||||
|
|
|
@ -72,7 +72,7 @@ struct _QtDemuxSample
|
||||||
guint32 size;
|
guint32 size;
|
||||||
guint64 offset;
|
guint64 offset;
|
||||||
guint64 timestamp; /* In GstClockTime */
|
guint64 timestamp; /* In GstClockTime */
|
||||||
guint32 duration; /* in stream->timescale units */
|
guint32 duration; /* in GstClockTime */
|
||||||
gboolean keyframe; /* TRUE when this packet is a keyframe */
|
gboolean keyframe; /* TRUE when this packet is a keyframe */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -106,20 +106,28 @@ struct _QtDemuxStream
|
||||||
QtDemuxSample *samples;
|
QtDemuxSample *samples;
|
||||||
gboolean all_keyframe; /* TRUE when all samples are keyframes (no stss) */
|
gboolean all_keyframe; /* TRUE when all samples are keyframes (no stss) */
|
||||||
|
|
||||||
|
/* if we use chunks or samples */
|
||||||
|
gboolean sampled;
|
||||||
|
|
||||||
|
/* video info */
|
||||||
gint width;
|
gint width;
|
||||||
gint height;
|
gint height;
|
||||||
/* Numerator/denominator framerate */
|
/* Numerator/denominator framerate */
|
||||||
gint fps_n;
|
gint fps_n;
|
||||||
gint fps_d;
|
gint fps_d;
|
||||||
|
|
||||||
gdouble rate;
|
|
||||||
gint n_channels;
|
|
||||||
guint bytes_per_frame;
|
|
||||||
guint compression;
|
|
||||||
guint samples_per_packet;
|
|
||||||
guint16 bits_per_sample;
|
guint16 bits_per_sample;
|
||||||
guint16 color_table_id;
|
guint16 color_table_id;
|
||||||
|
|
||||||
|
/* audio info */
|
||||||
|
gdouble rate;
|
||||||
|
gint n_channels;
|
||||||
|
guint samples_per_packet;
|
||||||
|
guint samples_per_frame;
|
||||||
|
guint bytes_per_packet;
|
||||||
|
guint bytes_per_sample;
|
||||||
|
guint bytes_per_frame;
|
||||||
|
guint compression;
|
||||||
|
|
||||||
/* when a discontinuity is pending */
|
/* when a discontinuity is pending */
|
||||||
gboolean discont;
|
gboolean discont;
|
||||||
|
|
||||||
|
@ -1154,19 +1162,8 @@ gst_qtdemux_prepare_current_sample (GstQTDemux * qtdemux,
|
||||||
|
|
||||||
*offset = sample->offset;
|
*offset = sample->offset;
|
||||||
*size = sample->size;
|
*size = sample->size;
|
||||||
|
*timestamp = sample->timestamp;
|
||||||
/* timestamps of AMR aren't known... FIXME */
|
*duration = sample->duration;
|
||||||
if (stream->fourcc == GST_MAKE_FOURCC ('s', 'a', 'm', 'r')) {
|
|
||||||
*duration = -1;
|
|
||||||
if (stream->sample_index == 0)
|
|
||||||
*timestamp = 0;
|
|
||||||
else
|
|
||||||
*timestamp = -1;
|
|
||||||
} else {
|
|
||||||
*timestamp = sample->timestamp;
|
|
||||||
*duration = gst_util_uint64_scale_int
|
|
||||||
(sample->duration, GST_SECOND, stream->timescale);
|
|
||||||
}
|
|
||||||
*keyframe = stream->all_keyframe || sample->keyframe;
|
*keyframe = stream->all_keyframe || sample->keyframe;
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
@ -1648,18 +1645,11 @@ gst_qtdemux_chain (GstPad * sinkpad, GstBuffer * inbuf)
|
||||||
GST_DEBUG_OBJECT (demux, "stream : %" GST_FOURCC_FORMAT,
|
GST_DEBUG_OBJECT (demux, "stream : %" GST_FOURCC_FORMAT,
|
||||||
GST_FOURCC_ARGS (stream->fourcc));
|
GST_FOURCC_ARGS (stream->fourcc));
|
||||||
|
|
||||||
if (stream->fourcc != GST_MAKE_FOURCC ('s', 'a', 'm', 'r')) {
|
GST_BUFFER_TIMESTAMP (outbuf) =
|
||||||
GST_BUFFER_TIMESTAMP (outbuf) =
|
stream->samples[stream->sample_index].timestamp;
|
||||||
stream->samples[stream->sample_index].timestamp;
|
demux->last_ts = GST_BUFFER_TIMESTAMP (outbuf);
|
||||||
demux->last_ts = GST_BUFFER_TIMESTAMP (outbuf);
|
GST_BUFFER_DURATION (outbuf) =
|
||||||
GST_BUFFER_DURATION (outbuf) = gst_util_uint64_scale_int
|
stream->samples[stream->sample_index].duration;
|
||||||
(stream->samples[stream->sample_index].duration, GST_SECOND,
|
|
||||||
stream->timescale);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
if (stream->sample_index == 0)
|
|
||||||
GST_BUFFER_TIMESTAMP (outbuf) = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* send buffer */
|
/* send buffer */
|
||||||
GST_LOG_OBJECT (demux,
|
GST_LOG_OBJECT (demux,
|
||||||
|
@ -1766,6 +1756,8 @@ gst_qtdemux_add_stream (GstQTDemux * qtdemux,
|
||||||
stream->pad =
|
stream->pad =
|
||||||
gst_pad_new_from_static_template (&gst_qtdemux_videosrc_template, name);
|
gst_pad_new_from_static_template (&gst_qtdemux_videosrc_template, name);
|
||||||
g_free (name);
|
g_free (name);
|
||||||
|
|
||||||
|
/* guess fps, FIXME */
|
||||||
if ((stream->n_samples == 1) && (stream->samples[0].duration == 0)) {
|
if ((stream->n_samples == 1) && (stream->samples[0].duration == 0)) {
|
||||||
stream->fps_n = 0;
|
stream->fps_n = 0;
|
||||||
stream->fps_d = 1;
|
stream->fps_d = 1;
|
||||||
|
@ -2720,9 +2712,8 @@ qtdemux_dump_stsz (GstQTDemux * qtdemux, void *buffer, int depth)
|
||||||
|
|
||||||
GST_LOG ("%*s version/flags: %08x", depth, "",
|
GST_LOG ("%*s version/flags: %08x", depth, "",
|
||||||
QTDEMUX_GUINT32_GET (buffer + 8));
|
QTDEMUX_GUINT32_GET (buffer + 8));
|
||||||
GST_LOG ("%*s sample size: %d", depth, "",
|
|
||||||
QTDEMUX_GUINT32_GET (buffer + 12));
|
|
||||||
sample_size = QTDEMUX_GUINT32_GET (buffer + 12);
|
sample_size = QTDEMUX_GUINT32_GET (buffer + 12);
|
||||||
|
GST_LOG ("%*s sample size: %d", depth, "", sample_size);
|
||||||
if (sample_size == 0) {
|
if (sample_size == 0) {
|
||||||
GST_LOG ("%*s n entries: %d", depth, "",
|
GST_LOG ("%*s n entries: %d", depth, "",
|
||||||
QTDEMUX_GUINT32_GET (buffer + 16));
|
QTDEMUX_GUINT32_GET (buffer + 16));
|
||||||
|
@ -2742,24 +2733,21 @@ qtdemux_dump_stsz (GstQTDemux * qtdemux, void *buffer, int depth)
|
||||||
static void
|
static void
|
||||||
qtdemux_dump_stco (GstQTDemux * qtdemux, void *buffer, int depth)
|
qtdemux_dump_stco (GstQTDemux * qtdemux, void *buffer, int depth)
|
||||||
{
|
{
|
||||||
//int i;
|
int i;
|
||||||
int n;
|
int n;
|
||||||
int offset;
|
int offset;
|
||||||
|
|
||||||
GST_LOG ("%*s version/flags: %08x", depth, "",
|
GST_LOG ("%*s version/flags: %08x", depth, "",
|
||||||
QTDEMUX_GUINT32_GET (buffer + 8));
|
QTDEMUX_GUINT32_GET (buffer + 8));
|
||||||
GST_LOG ("%*s n entries: %d", depth, "",
|
|
||||||
QTDEMUX_GUINT32_GET (buffer + 12));
|
|
||||||
n = QTDEMUX_GUINT32_GET (buffer + 12);
|
n = QTDEMUX_GUINT32_GET (buffer + 12);
|
||||||
|
GST_LOG ("%*s n entries: %d", depth, "", n);
|
||||||
offset = 16;
|
offset = 16;
|
||||||
#if 0
|
|
||||||
for (i = 0; i < n; i++) {
|
for (i = 0; i < n; i++) {
|
||||||
GST_LOG ("%*s chunk offset: %08x", depth, "",
|
GST_LOG ("%*s chunk offset: %d", depth, "",
|
||||||
QTDEMUX_GUINT32_GET (buffer + offset));
|
QTDEMUX_GUINT32_GET (buffer + offset));
|
||||||
|
|
||||||
offset += 4;
|
offset += 4;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -2965,7 +2953,7 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
int i, j, k;
|
int i, j, k;
|
||||||
QtDemuxStream *stream;
|
QtDemuxStream *stream;
|
||||||
int n_sample_times;
|
int n_sample_times;
|
||||||
guint64 timestamp;
|
guint64 timestamp, time;
|
||||||
int sample_size;
|
int sample_size;
|
||||||
int sample_index;
|
int sample_index;
|
||||||
GstTagList *list = NULL;
|
GstTagList *list = NULL;
|
||||||
|
@ -3137,6 +3125,7 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
int version, samplesize;
|
int version, samplesize;
|
||||||
guint32 fourcc;
|
guint32 fourcc;
|
||||||
int len;
|
int len;
|
||||||
|
guint16 compression_id;
|
||||||
|
|
||||||
len = QTDEMUX_GUINT32_GET (stsd->data + 16);
|
len = QTDEMUX_GUINT32_GET (stsd->data + 16);
|
||||||
GST_LOG ("st type: %" GST_FOURCC_FORMAT,
|
GST_LOG ("st type: %" GST_FOURCC_FORMAT,
|
||||||
|
@ -3145,61 +3134,75 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
stream->fourcc = fourcc = QTDEMUX_FOURCC_GET (stsd->data + 16 + 4);
|
stream->fourcc = fourcc = QTDEMUX_FOURCC_GET (stsd->data + 16 + 4);
|
||||||
offset = 32;
|
offset = 32;
|
||||||
|
|
||||||
GST_LOG ("version/rev: %08x",
|
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset));
|
|
||||||
version = QTDEMUX_GUINT32_GET (stsd->data + offset);
|
version = QTDEMUX_GUINT32_GET (stsd->data + offset);
|
||||||
|
stream->n_channels = QTDEMUX_GUINT16_GET (stsd->data + offset + 8);
|
||||||
|
samplesize = QTDEMUX_GUINT16_GET (stsd->data + offset + 10);
|
||||||
|
compression_id = QTDEMUX_GUINT16_GET (stsd->data + offset + 12);
|
||||||
|
stream->rate = QTDEMUX_FP32_GET (stsd->data + offset + 16);
|
||||||
|
|
||||||
|
GST_LOG ("version/rev: %08x", version);
|
||||||
GST_LOG ("vendor: %08x",
|
GST_LOG ("vendor: %08x",
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset + 4));
|
QTDEMUX_GUINT32_GET (stsd->data + offset + 4));
|
||||||
GST_LOG ("n_channels: %d",
|
GST_LOG ("n_channels: %d", stream->n_channels);
|
||||||
QTDEMUX_GUINT16_GET (stsd->data + offset + 8));
|
GST_LOG ("sample_size: %d", samplesize);
|
||||||
stream->n_channels = QTDEMUX_GUINT16_GET (stsd->data + offset + 8);
|
GST_LOG ("compression_id: %d", compression_id);
|
||||||
GST_LOG ("sample_size: %d",
|
|
||||||
QTDEMUX_GUINT16_GET (stsd->data + offset + 10));
|
|
||||||
samplesize = QTDEMUX_GUINT16_GET (stsd->data + offset + 10);
|
|
||||||
GST_LOG ("compression_id: %d",
|
|
||||||
QTDEMUX_GUINT16_GET (stsd->data + offset + 12));
|
|
||||||
GST_LOG ("packet size: %d",
|
GST_LOG ("packet size: %d",
|
||||||
QTDEMUX_GUINT16_GET (stsd->data + offset + 14));
|
QTDEMUX_GUINT16_GET (stsd->data + offset + 14));
|
||||||
GST_LOG ("sample rate: %g",
|
GST_LOG ("sample rate: %g", stream->rate);
|
||||||
QTDEMUX_FP32_GET (stsd->data + offset + 16));
|
|
||||||
stream->rate = QTDEMUX_FP32_GET (stsd->data + offset + 16);
|
if (compression_id == 0xfffe)
|
||||||
|
stream->sampled = TRUE;
|
||||||
|
|
||||||
offset = 52;
|
offset = 52;
|
||||||
if (version == 0x00010000) {
|
if (version == 0x00010000) {
|
||||||
GST_LOG ("samples/packet: %d",
|
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset));
|
|
||||||
stream->samples_per_packet = QTDEMUX_GUINT32_GET (stsd->data + offset);
|
stream->samples_per_packet = QTDEMUX_GUINT32_GET (stsd->data + offset);
|
||||||
GST_LOG ("bytes/packet: %d",
|
stream->samples_per_frame = stream->n_channels;
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset + 4));
|
stream->bytes_per_packet = QTDEMUX_GUINT32_GET (stsd->data + offset + 4);
|
||||||
GST_LOG ("bytes/frame: %d",
|
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset + 8));
|
|
||||||
stream->bytes_per_frame = QTDEMUX_GUINT32_GET (stsd->data + offset + 8);
|
stream->bytes_per_frame = QTDEMUX_GUINT32_GET (stsd->data + offset + 8);
|
||||||
GST_LOG ("bytes/sample: %d",
|
stream->bytes_per_sample = QTDEMUX_GUINT32_GET (stsd->data + offset + 12);
|
||||||
QTDEMUX_GUINT32_GET (stsd->data + offset + 12));
|
|
||||||
stream->compression = 1;
|
GST_LOG ("samples/packet: %d", stream->samples_per_packet);
|
||||||
|
GST_LOG ("bytes/packet: %d", stream->bytes_per_packet);
|
||||||
|
GST_LOG ("bytes/frame: %d", stream->bytes_per_frame);
|
||||||
|
GST_LOG ("bytes/sample: %d", stream->bytes_per_sample);
|
||||||
|
|
||||||
offset = 68;
|
offset = 68;
|
||||||
} else if (version == 0x00000000) {
|
} else if (version == 0x00000000) {
|
||||||
stream->bytes_per_frame = stream->n_channels * samplesize / 8;
|
stream->bytes_per_sample = samplesize / 8;
|
||||||
stream->samples_per_packet = 1;
|
stream->samples_per_frame = stream->n_channels;
|
||||||
stream->compression = 1;
|
stream->bytes_per_frame = stream->n_channels * stream->bytes_per_sample;
|
||||||
|
stream->samples_per_packet = stream->samples_per_frame;
|
||||||
|
stream->bytes_per_packet = stream->bytes_per_sample;
|
||||||
|
|
||||||
/* Yes, these have to be hard-coded */
|
/* Yes, these have to be hard-coded */
|
||||||
if (fourcc == GST_MAKE_FOURCC ('M', 'A', 'C', '6'))
|
if (fourcc == GST_MAKE_FOURCC ('M', 'A', 'C', '6')) {
|
||||||
stream->compression = 6;
|
stream->samples_per_packet = 6;
|
||||||
if (fourcc == GST_MAKE_FOURCC ('M', 'A', 'C', '3'))
|
stream->bytes_per_packet = 1;
|
||||||
stream->compression = 3;
|
stream->bytes_per_frame = 1 * stream->n_channels;
|
||||||
if (fourcc == GST_MAKE_FOURCC ('i', 'm', 'a', '4'))
|
stream->bytes_per_sample = 1;
|
||||||
stream->compression = 4;
|
stream->samples_per_frame = 6 * stream->n_channels;
|
||||||
if (fourcc == GST_MAKE_FOURCC ('s', 'a', 'm', 'r')) {
|
|
||||||
stream->n_channels = 1;
|
|
||||||
stream->rate = 8000;
|
|
||||||
stream->bytes_per_frame <<= 3;
|
|
||||||
}
|
}
|
||||||
if (fourcc == GST_MAKE_FOURCC ('u', 'l', 'a', 'w'))
|
if (fourcc == GST_MAKE_FOURCC ('M', 'A', 'C', '3')) {
|
||||||
stream->compression = 2;
|
stream->samples_per_packet = 3;
|
||||||
if (fourcc == GST_MAKE_FOURCC ('a', 'g', 's', 'm')) {
|
stream->bytes_per_packet = 1;
|
||||||
stream->bytes_per_frame *= 33;
|
stream->bytes_per_frame = 1 * stream->n_channels;
|
||||||
stream->compression = 320;
|
stream->bytes_per_sample = 1;
|
||||||
|
stream->samples_per_frame = 3 * stream->n_channels;
|
||||||
|
}
|
||||||
|
if (fourcc == GST_MAKE_FOURCC ('i', 'm', 'a', '4')) {
|
||||||
|
stream->samples_per_packet = 64;
|
||||||
|
stream->bytes_per_packet = 34;
|
||||||
|
stream->bytes_per_frame = 34 * stream->n_channels;
|
||||||
|
stream->bytes_per_sample = 2;
|
||||||
|
stream->samples_per_frame = 64 * stream->n_channels;
|
||||||
|
}
|
||||||
|
if (fourcc == GST_MAKE_FOURCC ('u', 'l', 'a', 'w') ||
|
||||||
|
fourcc == GST_MAKE_FOURCC ('a', 'l', 'a', 'w')) {
|
||||||
|
stream->samples_per_packet = 1;
|
||||||
|
stream->bytes_per_packet = 1;
|
||||||
|
stream->bytes_per_frame = 1 * stream->n_channels;
|
||||||
|
stream->bytes_per_sample = 2;
|
||||||
|
stream->samples_per_frame = 6 * stream->n_channels;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
GST_WARNING ("unknown version %08x", version);
|
GST_WARNING ("unknown version %08x", version);
|
||||||
|
@ -3281,6 +3284,12 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* promote to sampled format */
|
||||||
|
if (stream->fourcc == GST_MAKE_FOURCC ('s', 'a', 'm', 'r'))
|
||||||
|
stream->sampled = TRUE;
|
||||||
|
if (stream->fourcc == GST_MAKE_FOURCC ('m', 'p', '4', 'a'))
|
||||||
|
stream->sampled = TRUE;
|
||||||
|
|
||||||
/* sample to chunk */
|
/* sample to chunk */
|
||||||
stsc = qtdemux_tree_get_child_by_type (stbl, FOURCC_stsc);
|
stsc = qtdemux_tree_get_child_by_type (stbl, FOURCC_stsc);
|
||||||
g_assert (stsc);
|
g_assert (stsc);
|
||||||
|
@ -3298,7 +3307,7 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
stss = qtdemux_tree_get_child_by_type (stbl, FOURCC_stss);
|
stss = qtdemux_tree_get_child_by_type (stbl, FOURCC_stss);
|
||||||
|
|
||||||
sample_size = QTDEMUX_GUINT32_GET (stsz->data + 12);
|
sample_size = QTDEMUX_GUINT32_GET (stsz->data + 12);
|
||||||
if (sample_size == 0) {
|
if (sample_size == 0 || stream->sampled) {
|
||||||
n_samples = QTDEMUX_GUINT32_GET (stsz->data + 16);
|
n_samples = QTDEMUX_GUINT32_GET (stsz->data + 16);
|
||||||
GST_DEBUG_OBJECT (qtdemux, "stsz sample_size 0, allocating n_samples %d",
|
GST_DEBUG_OBJECT (qtdemux, "stsz sample_size 0, allocating n_samples %d",
|
||||||
n_samples);
|
n_samples);
|
||||||
|
@ -3307,7 +3316,11 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
stream->samples = samples;
|
stream->samples = samples;
|
||||||
|
|
||||||
for (i = 0; i < n_samples; i++) {
|
for (i = 0; i < n_samples; i++) {
|
||||||
samples[i].size = QTDEMUX_GUINT32_GET (stsz->data + i * 4 + 20);
|
if (sample_size == 0)
|
||||||
|
samples[i].size = QTDEMUX_GUINT32_GET (stsz->data + i * 4 + 20);
|
||||||
|
else
|
||||||
|
samples[i].size = sample_size;
|
||||||
|
|
||||||
GST_LOG_OBJECT (qtdemux, "sample %d has size %d", i, samples[i].size);
|
GST_LOG_OBJECT (qtdemux, "sample %d has size %d", i, samples[i].size);
|
||||||
/* init other fields to defaults for this sample */
|
/* init other fields to defaults for this sample */
|
||||||
samples[i].keyframe = FALSE;
|
samples[i].keyframe = FALSE;
|
||||||
|
@ -3351,6 +3364,7 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
|
|
||||||
n_sample_times = QTDEMUX_GUINT32_GET (stts->data + 12);
|
n_sample_times = QTDEMUX_GUINT32_GET (stts->data + 12);
|
||||||
timestamp = 0;
|
timestamp = 0;
|
||||||
|
time = 0;
|
||||||
index = 0;
|
index = 0;
|
||||||
for (i = 0; i < n_sample_times; i++) {
|
for (i = 0; i < n_sample_times; i++) {
|
||||||
guint32 n;
|
guint32 n;
|
||||||
|
@ -3359,18 +3373,16 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
n = QTDEMUX_GUINT32_GET (stts->data + 16 + 8 * i);
|
n = QTDEMUX_GUINT32_GET (stts->data + 16 + 8 * i);
|
||||||
duration = QTDEMUX_GUINT32_GET (stts->data + 16 + 8 * i + 4);
|
duration = QTDEMUX_GUINT32_GET (stts->data + 16 + 8 * i + 4);
|
||||||
for (j = 0; j < n; j++) {
|
for (j = 0; j < n; j++) {
|
||||||
guint64 time;
|
|
||||||
|
|
||||||
time = gst_util_uint64_scale_int (timestamp,
|
|
||||||
GST_SECOND, stream->timescale);
|
|
||||||
|
|
||||||
GST_INFO_OBJECT (qtdemux, "sample %d: timestamp %" GST_TIME_FORMAT,
|
GST_INFO_OBJECT (qtdemux, "sample %d: timestamp %" GST_TIME_FORMAT,
|
||||||
index, GST_TIME_ARGS (time));
|
index, GST_TIME_ARGS (timestamp));
|
||||||
|
|
||||||
samples[index].timestamp = time;
|
samples[index].timestamp = timestamp;
|
||||||
samples[index].duration = duration;
|
|
||||||
/* add non-scaled values to avoid rounding errors */
|
/* add non-scaled values to avoid rounding errors */
|
||||||
timestamp += duration;
|
time += duration;
|
||||||
|
timestamp = gst_util_uint64_scale_int (time,
|
||||||
|
GST_SECOND, stream->timescale);
|
||||||
|
samples[index].duration = timestamp - samples[index].timestamp;
|
||||||
|
|
||||||
index++;
|
index++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3395,8 +3407,6 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
stream->all_keyframe = TRUE;
|
stream->all_keyframe = TRUE;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
guint64 timestamp = 0;
|
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (qtdemux,
|
GST_DEBUG_OBJECT (qtdemux,
|
||||||
"stsz sample_size %d != 0, treating chunks as samples", sample_size);
|
"stsz sample_size %d != 0, treating chunks as samples", sample_size);
|
||||||
|
|
||||||
|
@ -3415,16 +3425,17 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
GST_DEBUG_OBJECT (qtdemux, "n_samples_per_chunk %d", n_samples_per_chunk);
|
GST_DEBUG_OBJECT (qtdemux, "n_samples_per_chunk %d", n_samples_per_chunk);
|
||||||
offset = 16;
|
offset = 16;
|
||||||
sample_index = 0;
|
sample_index = 0;
|
||||||
|
timestamp = 0;
|
||||||
for (i = 0; i < n_samples_per_chunk; i++) {
|
for (i = 0; i < n_samples_per_chunk; i++) {
|
||||||
int first_chunk, last_chunk;
|
guint32 first_chunk, last_chunk;
|
||||||
int samples_per_chunk;
|
guint32 samples_per_chunk;
|
||||||
|
|
||||||
first_chunk = QTDEMUX_GUINT32_GET (stsc->data + 16 + i * 12 + 0) - 1;
|
first_chunk = QTDEMUX_GUINT32_GET (stsc->data + 16 + i * 12 + 0) - 1;
|
||||||
/* the last chunk of each entry is calculated by taking the first chunk
|
/* the last chunk of each entry is calculated by taking the first chunk
|
||||||
* of the next entry; except if there is no next, where we fake it with
|
* of the next entry; except if there is no next, where we fake it with
|
||||||
* INT_MAX */
|
* INT_MAX */
|
||||||
if (i == n_samples_per_chunk - 1) {
|
if (i == n_samples_per_chunk - 1) {
|
||||||
last_chunk = INT_MAX;
|
last_chunk = G_MAXUINT32;
|
||||||
} else {
|
} else {
|
||||||
last_chunk = QTDEMUX_GUINT32_GET (stsc->data + 16 + i * 12 + 12) - 1;
|
last_chunk = QTDEMUX_GUINT32_GET (stsc->data + 16 + i * 12 + 12) - 1;
|
||||||
}
|
}
|
||||||
|
@ -3441,35 +3452,30 @@ qtdemux_parse_trak (GstQTDemux * qtdemux, GNode * trak)
|
||||||
goto done2;
|
goto done2;
|
||||||
if (stco) {
|
if (stco) {
|
||||||
chunk_offset = QTDEMUX_GUINT32_GET (stco->data + 16 + j * 4);
|
chunk_offset = QTDEMUX_GUINT32_GET (stco->data + 16 + j * 4);
|
||||||
GST_LOG_OBJECT (qtdemux, "stco chunk %d offset %x", j, chunk_offset);
|
|
||||||
} else {
|
} else {
|
||||||
chunk_offset = QTDEMUX_GUINT64_GET (co64->data + 16 + j * 8);
|
chunk_offset = QTDEMUX_GUINT64_GET (co64->data + 16 + j * 8);
|
||||||
GST_LOG_OBJECT (qtdemux, "co64 chunk %d offset %" G_GUINT64_FORMAT, j,
|
|
||||||
chunk_offset);
|
|
||||||
}
|
}
|
||||||
GST_LOG_OBJECT (qtdemux, "Creating entry %d with offset %lld",
|
GST_LOG_OBJECT (qtdemux,
|
||||||
j, chunk_offset);
|
"Creating entry %d with offset %" G_GUINT64_FORMAT, j,
|
||||||
|
chunk_offset);
|
||||||
|
|
||||||
samples[j].chunk = j;
|
samples[j].chunk = j;
|
||||||
samples[j].offset = chunk_offset;
|
samples[j].offset = chunk_offset;
|
||||||
if (stream->samples_per_packet * stream->compression != 0)
|
|
||||||
samples[j].size =
|
|
||||||
samples_per_chunk * stream->bytes_per_frame /
|
|
||||||
stream->samples_per_packet / stream->compression;
|
|
||||||
else if (stream->bytes_per_frame)
|
|
||||||
samples[j].size = stream->bytes_per_frame;
|
|
||||||
else
|
|
||||||
samples[j].size = sample_size;
|
|
||||||
/* FIXME, divide by 2? must be sample size or something.. */
|
|
||||||
samples[j].duration =
|
|
||||||
samples_per_chunk * stream->timescale / (stream->rate / 2);
|
|
||||||
samples[j].timestamp = timestamp;
|
|
||||||
samples[j].keyframe = TRUE;
|
|
||||||
|
|
||||||
if (stream->rate > 0) {
|
samples[j].size = (samples_per_chunk * stream->n_channels) /
|
||||||
timestamp += gst_util_uint64_scale_int (samples_per_chunk,
|
stream->samples_per_frame * stream->bytes_per_frame;
|
||||||
GST_SECOND, stream->rate);
|
|
||||||
}
|
GST_INFO_OBJECT (qtdemux, "sample %d: timestamp %" GST_TIME_FORMAT
|
||||||
|
", size %u", j, GST_TIME_ARGS (timestamp), samples[j].size);
|
||||||
|
|
||||||
|
samples[j].timestamp = timestamp;
|
||||||
sample_index += samples_per_chunk;
|
sample_index += samples_per_chunk;
|
||||||
|
|
||||||
|
timestamp = gst_util_uint64_scale_int (sample_index,
|
||||||
|
GST_SECOND, stream->timescale);
|
||||||
|
samples[j].duration = timestamp - samples[j].timestamp;
|
||||||
|
|
||||||
|
samples[j].keyframe = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3477,6 +3483,8 @@ done2:
|
||||||
|
|
||||||
/* parse and prepare segment info from the edit list */
|
/* parse and prepare segment info from the edit list */
|
||||||
GST_DEBUG_OBJECT (qtdemux, "looking for edit list container");
|
GST_DEBUG_OBJECT (qtdemux, "looking for edit list container");
|
||||||
|
stream->n_segments = 0;
|
||||||
|
stream->segments = NULL;
|
||||||
if ((edts = qtdemux_tree_get_child_by_type (trak, FOURCC_edts))) {
|
if ((edts = qtdemux_tree_get_child_by_type (trak, FOURCC_edts))) {
|
||||||
GNode *elst;
|
GNode *elst;
|
||||||
gint n_segments;
|
gint n_segments;
|
||||||
|
@ -3534,24 +3542,27 @@ done2:
|
||||||
GST_TIME_ARGS (segment->media_start), segment->rate);
|
GST_TIME_ARGS (segment->media_start), segment->rate);
|
||||||
}
|
}
|
||||||
GST_DEBUG_OBJECT (qtdemux, "found %d non-empty segments", count);
|
GST_DEBUG_OBJECT (qtdemux, "found %d non-empty segments", count);
|
||||||
|
|
||||||
if (count == 0) {
|
|
||||||
/* no segments, create one to play the complete trak */
|
|
||||||
count = 1;
|
|
||||||
stream->segments[0].time = 0;
|
|
||||||
stream->segments[0].stop_time = stream->duration;
|
|
||||||
stream->segments[0].duration = stream->duration;
|
|
||||||
stream->segments[0].media_start = 0;
|
|
||||||
stream->segments[0].media_stop = stream->duration;
|
|
||||||
stream->segments[0].rate = 1.0;
|
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (qtdemux, "created dummy segment");
|
|
||||||
}
|
|
||||||
GST_DEBUG_OBJECT (qtdemux, "using %d segments", count);
|
|
||||||
stream->n_segments = count;
|
stream->n_segments = count;
|
||||||
}
|
}
|
||||||
done3:
|
done3:
|
||||||
|
|
||||||
|
/* no segments, create one to play the complete trak */
|
||||||
|
if (stream->n_segments == 0) {
|
||||||
|
if (stream->segments == NULL)
|
||||||
|
stream->segments = g_new (QtDemuxSegment, 1);
|
||||||
|
|
||||||
|
stream->segments[0].time = 0;
|
||||||
|
stream->segments[0].stop_time = qtdemux->segment.duration;
|
||||||
|
stream->segments[0].duration = qtdemux->segment.duration;
|
||||||
|
stream->segments[0].media_start = 0;
|
||||||
|
stream->segments[0].media_stop = qtdemux->segment.duration;
|
||||||
|
stream->segments[0].rate = 1.0;
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (qtdemux, "created dummy segment");
|
||||||
|
stream->n_segments = 1;
|
||||||
|
}
|
||||||
|
GST_DEBUG_OBJECT (qtdemux, "using %d segments", stream->n_segments);
|
||||||
|
|
||||||
gst_qtdemux_add_stream (qtdemux, stream, list);
|
gst_qtdemux_add_stream (qtdemux, stream, list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue