diff --git a/doc/ffmpeg-doc.texi b/doc/ffmpeg-doc.texi index e2e0df7f2f8946645a1cc1e29f9b266ee265a99b..ed46b49c0084f599cb36aa7baaa12a32f19be8bf 100644 --- a/doc/ffmpeg-doc.texi +++ b/doc/ffmpeg-doc.texi @@ -630,6 +630,10 @@ of the output file: @example ffmpeg -i in.ogg -map_meta_data 0:0,s0 out.mp3 @end example +@item -map_chapters @var{outfile}:@var{infile} +Copy chapters from @var{infile} to @var{outfile}. If no chapter mapping is specified, +then chapters are copied from the first input file with at least one chapter to all +output files. Use a negative file index to disable any chapter copying. @item -debug Print specific debug info. @item -benchmark diff --git a/ffmpeg.c b/ffmpeg.c index b1cdf1c7be54a78bdf592f22144fe37392b67d92..9b2cc20b459fb71dea9e2d0cc2037d16cd44c84d 100644 --- a/ffmpeg.c +++ b/ffmpeg.c @@ -102,6 +102,11 @@ typedef struct AVMetaDataMap { int index; //< stream/chapter/program number } AVMetaDataMap; +typedef struct AVChapterMap { + int in_file; + int out_file; +} AVChapterMap; + static const OptionDef options[]; #define MAX_FILES 100 @@ -132,6 +137,9 @@ static int nb_meta_data_maps; static int metadata_streams_autocopy = 1; static int metadata_chapters_autocopy = 1; +static AVChapterMap *chapter_maps = NULL; +static int nb_chapter_maps; + /* indexed by output file stream index */ static int *streamid_map = NULL; static int nb_streamid_map = 0; @@ -2381,7 +2389,28 @@ static int transcode(AVFormatContext **output_files, av_metadata_set2(meta[0], mtag->key, mtag->value, AV_METADATA_DONT_OVERWRITE); } + /* copy chapters according to chapter maps */ + for (i = 0; i < nb_chapter_maps; i++) { + int infile = chapter_maps[i].in_file; + int outfile = chapter_maps[i].out_file; + + if (infile < 0 || outfile < 0) + continue; + if (infile >= nb_input_files) { + snprintf(error, sizeof(error), "Invalid input file index %d in chapter mapping.\n", infile); + ret = AVERROR(EINVAL); + goto dump_format; + } + if (outfile >= nb_output_files) { + snprintf(error, sizeof(error), "Invalid output file index %d in chapter mapping.\n",outfile); + ret = AVERROR(EINVAL); + goto dump_format; + } + copy_chapters(infile, outfile); + } + /* copy chapters from the first input file that has them*/ + if (!nb_chapter_maps) for (i = 0; i < nb_input_files; i++) { if (!input_files[i]->nb_chapters) continue; @@ -2962,6 +2991,21 @@ static void opt_map_meta_data(const char *arg) metadata_chapters_autocopy = 0; } +static void opt_map_chapters(const char *arg) +{ + AVChapterMap *c; + char *p; + + chapter_maps = grow_array(chapter_maps, sizeof(*chapter_maps), &nb_chapter_maps, + nb_chapter_maps + 1); + c = &chapter_maps[nb_chapter_maps - 1]; + c->out_file = strtol(arg, &p, 0); + if (*p) + p++; + + c->in_file = strtol(p, &p, 0); +} + static void opt_input_ts_scale(const char *arg) { unsigned int stream; @@ -4074,6 +4118,7 @@ static const OptionDef options[] = { { "y", OPT_BOOL, {(void*)&file_overwrite}, "overwrite output files" }, { "map", HAS_ARG | OPT_EXPERT, {(void*)opt_map}, "set input stream mapping", "file:stream[:syncfile:syncstream]" }, { "map_meta_data", HAS_ARG | OPT_EXPERT, {(void*)opt_map_meta_data}, "set meta data information of outfile from infile", "outfile[,metadata]:infile[,metadata]" }, + { "map_chapters", HAS_ARG | OPT_EXPERT, {(void*)opt_map_chapters}, "set chapters mapping", "outfile:infile" }, { "t", OPT_FUNC2 | HAS_ARG, {(void*)opt_recording_time}, "record or transcode \"duration\" seconds of audio/video", "duration" }, { "fs", HAS_ARG | OPT_INT64, {(void*)&limit_filesize}, "set the limit file size in bytes", "limit_size" }, // { "ss", OPT_FUNC2 | HAS_ARG, {(void*)opt_start_time}, "set the start time offset", "time_off" },