lavf/concatdec: allow to match streams by id.

That makes the concat demuxer usable with MPEG-PS streams,
even when the streams in the different parts are detected
in different order.
pull/64/head
Nicolas George 11 years ago
parent cc6d549adb
commit 26dea7731e
  1. 18
      doc/demuxers.texi
  2. 94
      libavformat/concatdec.c

@ -74,7 +74,7 @@ following directive is recognized:
Path to a file to read; special characters and spaces must be escaped with Path to a file to read; special characters and spaces must be escaped with
backslash or single quotes. backslash or single quotes.
All subsequent directives apply to that file. All subsequent file-related directives apply to that file.
@item @code{ffconcat version 1.0} @item @code{ffconcat version 1.0}
Identify the script type and version. It also sets the @option{safe} option Identify the script type and version. It also sets the @option{safe} option
@ -92,6 +92,22 @@ file is not available or accurate.
If the duration is set for all files, then it is possible to seek in the If the duration is set for all files, then it is possible to seek in the
whole concatenated video. whole concatenated video.
@item @code{stream}
Introduce a stream in the virtual file.
All subsequent stream-related directives apply to the last introduced
stream.
Some streams properties must be set in order to allow identifying the
matching streams in the subfiles.
If no streams are defined in the script, the streams from the first file are
copied.
@item @code{exact_stream_id @var{id}}
Set the id of the stream.
If this directive is given, the string with the corresponding id in the
subfiles will be used.
This is especially useful for MPEG-PS (VOB) files, where the order of the
streams is not reliable.
@end table @end table
@subsection Options @subsection Options

@ -29,6 +29,8 @@ typedef struct {
char *url; char *url;
int64_t start_time; int64_t start_time;
int64_t duration; int64_t duration;
int *stream_map;
int stream_map_size;
} ConcatFile; } ConcatFile;
typedef struct { typedef struct {
@ -39,6 +41,7 @@ typedef struct {
AVFormatContext *avf; AVFormatContext *avf;
int safe; int safe;
int seekable; int seekable;
int match_streams;
} ConcatContext; } ConcatContext;
static int concat_probe(AVProbeData *probe) static int concat_probe(AVProbeData *probe)
@ -134,6 +137,54 @@ fail:
return ret; return ret;
} }
static int copy_stream_props(AVStream *st, AVStream *source_st)
{
int ret;
if ((ret = avcodec_copy_context(st->codec, source_st->codec)) < 0)
return ret;
st->r_frame_rate = source_st->r_frame_rate;
st->avg_frame_rate = source_st->avg_frame_rate;
st->time_base = source_st->time_base;
st->sample_aspect_ratio = source_st->sample_aspect_ratio;
return 0;
}
static int match_streams(AVFormatContext *avf)
{
ConcatContext *cat = avf->priv_data;
AVStream *st;
int *map, i, j, ret;
if (!cat->match_streams ||
cat->cur_file->stream_map_size >= cat->avf->nb_streams)
return 0;
map = av_realloc(cat->cur_file->stream_map,
cat->avf->nb_streams * sizeof(*map));
if (!map)
return AVERROR(ENOMEM);
cat->cur_file->stream_map = map;
for (i = cat->cur_file->stream_map_size; i < cat->avf->nb_streams; i++) {
st = cat->avf->streams[i];
map[i] = -1;
for (j = 0; j < avf->nb_streams; j++) {
if (avf->streams[j]->id == st->id) {
av_log(avf, AV_LOG_VERBOSE,
"Match slave stream #%d with stream #%d id 0x%x\n",
i, j, st->id);
map[i] = j;
if (!avf->streams[j]->codec->codec_id && st->codec->codec_id)
if ((ret = copy_stream_props(avf->streams[j], st)) < 0)
return ret;
}
}
}
cat->cur_file->stream_map_size = cat->avf->nb_streams;
return 0;
}
static int open_file(AVFormatContext *avf, unsigned fileno) static int open_file(AVFormatContext *avf, unsigned fileno)
{ {
ConcatContext *cat = avf->priv_data; ConcatContext *cat = avf->priv_data;
@ -159,6 +210,8 @@ static int open_file(AVFormatContext *avf, unsigned fileno)
file->start_time = !fileno ? 0 : file->start_time = !fileno ? 0 :
cat->files[fileno - 1].start_time + cat->files[fileno - 1].start_time +
cat->files[fileno - 1].duration; cat->files[fileno - 1].duration;
if ((ret = match_streams(avf)) < 0)
return ret;
return 0; return 0;
} }
@ -183,7 +236,7 @@ static int concat_read_header(AVFormatContext *avf)
int ret, line = 0, i; int ret, line = 0, i;
unsigned nb_files_alloc = 0; unsigned nb_files_alloc = 0;
ConcatFile *file = NULL; ConcatFile *file = NULL;
AVStream *st, *source_st; AVStream *st;
int64_t time = 0; int64_t time = 0;
while (1) { while (1) {
@ -217,6 +270,17 @@ static int concat_read_header(AVFormatContext *avf)
FAIL(ret); FAIL(ret);
} }
file->duration = dur; file->duration = dur;
} else if (!strcmp(keyword, "stream")) {
if (!avformat_new_stream(avf, NULL))
FAIL(AVERROR(ENOMEM));
} else if (!strcmp(keyword, "exact_stream_id")) {
if (!avf->nb_streams) {
av_log(avf, AV_LOG_ERROR, "Line %d: exact_stream_id without stream\n",
line);
FAIL(AVERROR_INVALIDDATA);
}
avf->streams[avf->nb_streams - 1]->id =
strtol(get_keyword(&cursor), NULL, 0);
} else if (!strcmp(keyword, "ffconcat")) { } else if (!strcmp(keyword, "ffconcat")) {
char *ver_kw = get_keyword(&cursor); char *ver_kw = get_keyword(&cursor);
char *ver_val = get_keyword(&cursor); char *ver_val = get_keyword(&cursor);
@ -251,18 +315,16 @@ static int concat_read_header(AVFormatContext *avf)
cat->seekable = 1; cat->seekable = 1;
} }
cat->match_streams = !!avf->nb_streams;
if ((ret = open_file(avf, 0)) < 0) if ((ret = open_file(avf, 0)) < 0)
FAIL(ret); FAIL(ret);
if (!cat->match_streams) {
for (i = 0; i < cat->avf->nb_streams; i++) { for (i = 0; i < cat->avf->nb_streams; i++) {
if (!(st = avformat_new_stream(avf, NULL))) if (!(st = avformat_new_stream(avf, NULL)))
FAIL(AVERROR(ENOMEM)); FAIL(AVERROR(ENOMEM));
source_st = cat->avf->streams[i]; if ((ret = copy_stream_props(st, cat->avf->streams[i])) < 0)
if ((ret = avcodec_copy_context(st->codec, source_st->codec)) < 0)
FAIL(ret); FAIL(ret);
st->r_frame_rate = source_st->r_frame_rate; }
st->avg_frame_rate = source_st->avg_frame_rate;
st->time_base = source_st->time_base;
st->sample_aspect_ratio = source_st->sample_aspect_ratio;
} }
return 0; return 0;
@ -292,12 +354,24 @@ static int concat_read_packet(AVFormatContext *avf, AVPacket *pkt)
int64_t delta; int64_t delta;
while (1) { while (1) {
if ((ret = av_read_frame(cat->avf, pkt)) != AVERROR_EOF || ret = av_read_frame(cat->avf, pkt);
(ret = open_next_file(avf)) < 0) if (ret == AVERROR_EOF) {
break; if ((ret = open_next_file(avf)) < 0)
return ret;
continue;
} }
if (ret < 0) if (ret < 0)
return ret; return ret;
if (cat->match_streams) {
match_streams(avf);
pkt->stream_index = cat->cur_file->stream_map[pkt->stream_index];
if (pkt->stream_index < 0) {
av_packet_unref(pkt);
continue;
}
}
break;
}
delta = av_rescale_q(cat->cur_file->start_time - cat->avf->start_time, delta = av_rescale_q(cat->cur_file->start_time - cat->avf->start_time,
AV_TIME_BASE_Q, AV_TIME_BASE_Q,

Loading…
Cancel
Save