You've already forked FFmpeg
mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-08-15 14:13:16 +02:00
libavformat/mxfdec: don't assume first stream index to be primary
Signed-off-by: Michael Niedermayer <michael@niedermayer.cc>
This commit is contained in:
committed by
Michael Niedermayer
parent
3b82be9e3b
commit
263f8fd7e5
@@ -1491,6 +1491,14 @@ static int mxf_compute_index_tables(MXFContext *mxf)
|
|||||||
{
|
{
|
||||||
int i, j, k, ret, nb_sorted_segments;
|
int i, j, k, ret, nb_sorted_segments;
|
||||||
MXFIndexTableSegment **sorted_segments = NULL;
|
MXFIndexTableSegment **sorted_segments = NULL;
|
||||||
|
AVStream *st = NULL;
|
||||||
|
|
||||||
|
for (i = 0; i < mxf->fc->nb_streams; i++) {
|
||||||
|
if (mxf->fc->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_DATA)
|
||||||
|
continue;
|
||||||
|
st = mxf->fc->streams[i];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if ((ret = mxf_get_sorted_table_segments(mxf, &nb_sorted_segments, &sorted_segments)) ||
|
if ((ret = mxf_get_sorted_table_segments(mxf, &nb_sorted_segments, &sorted_segments)) ||
|
||||||
nb_sorted_segments <= 0) {
|
nb_sorted_segments <= 0) {
|
||||||
@@ -1560,7 +1568,7 @@ static int mxf_compute_index_tables(MXFContext *mxf)
|
|||||||
av_log(mxf->fc, AV_LOG_WARNING, "IndexSID %i segment %i has zero IndexDuration and there's more than one segment\n",
|
av_log(mxf->fc, AV_LOG_WARNING, "IndexSID %i segment %i has zero IndexDuration and there's more than one segment\n",
|
||||||
t->index_sid, k);
|
t->index_sid, k);
|
||||||
|
|
||||||
if (mxf->fc->nb_streams <= 0) {
|
if (!st) {
|
||||||
av_log(mxf->fc, AV_LOG_WARNING, "no streams?\n");
|
av_log(mxf->fc, AV_LOG_WARNING, "no streams?\n");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -1568,7 +1576,7 @@ static int mxf_compute_index_tables(MXFContext *mxf)
|
|||||||
/* assume the first stream's duration is reasonable
|
/* assume the first stream's duration is reasonable
|
||||||
* leave index_duration = 0 on further segments in case we have any (unlikely)
|
* leave index_duration = 0 on further segments in case we have any (unlikely)
|
||||||
*/
|
*/
|
||||||
t->segments[k]->index_duration = mxf->fc->streams[0]->duration;
|
t->segments[k]->index_duration = st->duration;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2617,6 +2625,19 @@ static int is_pcm(enum AVCodecID codec_id)
|
|||||||
return codec_id >= AV_CODEC_ID_PCM_S16LE && codec_id < AV_CODEC_ID_PCM_S24DAUD;
|
return codec_id >= AV_CODEC_ID_PCM_S16LE && codec_id < AV_CODEC_ID_PCM_S24DAUD;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static AVStream* mxf_get_opatom_stream(MXFContext *mxf)
|
||||||
|
{
|
||||||
|
if (mxf->op != OPAtom)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
for (int i = 0; i < mxf->fc->nb_streams; i++) {
|
||||||
|
if (mxf->fc->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_DATA)
|
||||||
|
continue;
|
||||||
|
return mxf->fc->streams[i];
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Deal with the case where for some audio atoms EditUnitByteCount is
|
* Deal with the case where for some audio atoms EditUnitByteCount is
|
||||||
* very small (2, 4..). In those cases we should read more than one
|
* very small (2, 4..). In those cases we should read more than one
|
||||||
@@ -2628,13 +2649,13 @@ static void mxf_handle_small_eubc(AVFormatContext *s)
|
|||||||
|
|
||||||
/* assuming non-OPAtom == frame wrapped
|
/* assuming non-OPAtom == frame wrapped
|
||||||
* no sane writer would wrap 2 byte PCM packets with 20 byte headers.. */
|
* no sane writer would wrap 2 byte PCM packets with 20 byte headers.. */
|
||||||
if (mxf->op != OPAtom)
|
AVStream *st = mxf_get_opatom_stream(mxf);
|
||||||
|
if (!st)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* expect PCM with exactly one index table segment and a small (< 32) EUBC */
|
/* expect PCM with exactly one index table segment and a small (< 32) EUBC */
|
||||||
if (s->nb_streams != 1 ||
|
if (st->codecpar->codec_type != AVMEDIA_TYPE_AUDIO ||
|
||||||
s->streams[0]->codecpar->codec_type != AVMEDIA_TYPE_AUDIO ||
|
!is_pcm(st->codecpar->codec_id) ||
|
||||||
!is_pcm(s->streams[0]->codecpar->codec_id) ||
|
|
||||||
mxf->nb_index_tables != 1 ||
|
mxf->nb_index_tables != 1 ||
|
||||||
mxf->index_tables[0].nb_segments != 1 ||
|
mxf->index_tables[0].nb_segments != 1 ||
|
||||||
mxf->index_tables[0].segments[0]->edit_unit_byte_count >= 32)
|
mxf->index_tables[0].segments[0]->edit_unit_byte_count >= 32)
|
||||||
@@ -2659,11 +2680,12 @@ static int mxf_handle_missing_index_segment(MXFContext *mxf)
|
|||||||
int essence_partition_count = 0;
|
int essence_partition_count = 0;
|
||||||
int i, ret;
|
int i, ret;
|
||||||
|
|
||||||
if (mxf->op != OPAtom)
|
st = mxf_get_opatom_stream(mxf);
|
||||||
|
if (!st)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* TODO: support raw video without an index if they exist */
|
/* TODO: support raw video without an index if they exist */
|
||||||
if (s->nb_streams != 1 || s->streams[0]->codecpar->codec_type != AVMEDIA_TYPE_AUDIO || !is_pcm(s->streams[0]->codecpar->codec_id))
|
if (st->codecpar->codec_type != AVMEDIA_TYPE_AUDIO || !is_pcm(st->codecpar->codec_id))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* check if file already has a IndexTableSegment */
|
/* check if file already has a IndexTableSegment */
|
||||||
@@ -2694,7 +2716,6 @@ static int mxf_handle_missing_index_segment(MXFContext *mxf)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
st = s->streams[0];
|
|
||||||
segment->type = IndexTableSegment;
|
segment->type = IndexTableSegment;
|
||||||
/* stream will be treated as small EditUnitByteCount */
|
/* stream will be treated as small EditUnitByteCount */
|
||||||
segment->edit_unit_byte_count = (av_get_bits_per_sample(st->codecpar->codec_id) * st->codecpar->channels) >> 3;
|
segment->edit_unit_byte_count = (av_get_bits_per_sample(st->codecpar->codec_id) * st->codecpar->channels) >> 3;
|
||||||
@@ -3105,12 +3126,12 @@ static int mxf_read_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
return mxf_read_packet_old(s, pkt);
|
return mxf_read_packet_old(s, pkt);
|
||||||
|
|
||||||
// If we have no streams then we basically are at EOF
|
// If we have no streams then we basically are at EOF
|
||||||
if (s->nb_streams < 1)
|
st = mxf_get_opatom_stream(mxf);
|
||||||
|
if (!st)
|
||||||
return AVERROR_EOF;
|
return AVERROR_EOF;
|
||||||
|
|
||||||
/* OPAtom - clip wrapped demuxing */
|
/* OPAtom - clip wrapped demuxing */
|
||||||
/* NOTE: mxf_read_header() makes sure nb_index_tables > 0 for OPAtom */
|
/* NOTE: mxf_read_header() makes sure nb_index_tables > 0 for OPAtom */
|
||||||
st = s->streams[0];
|
|
||||||
t = &mxf->index_tables[0];
|
t = &mxf->index_tables[0];
|
||||||
|
|
||||||
if (mxf->current_edit_unit >= st->duration)
|
if (mxf->current_edit_unit >= st->duration)
|
||||||
@@ -3140,7 +3161,7 @@ static int mxf_read_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
if ((size = av_get_packet(s->pb, pkt, size)) < 0)
|
if ((size = av_get_packet(s->pb, pkt, size)) < 0)
|
||||||
return size;
|
return size;
|
||||||
|
|
||||||
pkt->stream_index = 0;
|
pkt->stream_index = st->index;
|
||||||
|
|
||||||
if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO && t->ptses &&
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO && t->ptses &&
|
||||||
mxf->current_edit_unit >= 0 && mxf->current_edit_unit < t->nb_ptses) {
|
mxf->current_edit_unit >= 0 && mxf->current_edit_unit < t->nb_ptses) {
|
||||||
@@ -3225,6 +3246,9 @@ static int mxf_read_seek(AVFormatContext *s, int stream_index, int64_t sample_ti
|
|||||||
MXFIndexTable *t;
|
MXFIndexTable *t;
|
||||||
MXFTrack *source_track = st->priv_data;
|
MXFTrack *source_track = st->priv_data;
|
||||||
|
|
||||||
|
if(st->codecpar->codec_type == AVMEDIA_TYPE_DATA)
|
||||||
|
return 0;
|
||||||
|
|
||||||
/* if audio then truncate sample_time to EditRate */
|
/* if audio then truncate sample_time to EditRate */
|
||||||
if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO)
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO)
|
||||||
sample_time = av_rescale_q(sample_time, st->time_base,
|
sample_time = av_rescale_q(sample_time, st->time_base,
|
||||||
|
Reference in New Issue
Block a user