mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2024-12-23 12:43:46 +02:00
ffmpeg.c: add an option for copying chapters
fixes issue 2068 Originally committed as revision 25727 to svn://svn.ffmpeg.org/ffmpeg/trunk
This commit is contained in:
parent
5fb2f1347e
commit
91e96ebafb
@ -630,6 +630,10 @@ of the output file:
|
|||||||
@example
|
@example
|
||||||
ffmpeg -i in.ogg -map_meta_data 0:0,s0 out.mp3
|
ffmpeg -i in.ogg -map_meta_data 0:0,s0 out.mp3
|
||||||
@end example
|
@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
|
@item -debug
|
||||||
Print specific debug info.
|
Print specific debug info.
|
||||||
@item -benchmark
|
@item -benchmark
|
||||||
|
45
ffmpeg.c
45
ffmpeg.c
@ -102,6 +102,11 @@ typedef struct AVMetaDataMap {
|
|||||||
int index; //< stream/chapter/program number
|
int index; //< stream/chapter/program number
|
||||||
} AVMetaDataMap;
|
} AVMetaDataMap;
|
||||||
|
|
||||||
|
typedef struct AVChapterMap {
|
||||||
|
int in_file;
|
||||||
|
int out_file;
|
||||||
|
} AVChapterMap;
|
||||||
|
|
||||||
static const OptionDef options[];
|
static const OptionDef options[];
|
||||||
|
|
||||||
#define MAX_FILES 100
|
#define MAX_FILES 100
|
||||||
@ -132,6 +137,9 @@ static int nb_meta_data_maps;
|
|||||||
static int metadata_streams_autocopy = 1;
|
static int metadata_streams_autocopy = 1;
|
||||||
static int metadata_chapters_autocopy = 1;
|
static int metadata_chapters_autocopy = 1;
|
||||||
|
|
||||||
|
static AVChapterMap *chapter_maps = NULL;
|
||||||
|
static int nb_chapter_maps;
|
||||||
|
|
||||||
/* indexed by output file stream index */
|
/* indexed by output file stream index */
|
||||||
static int *streamid_map = NULL;
|
static int *streamid_map = NULL;
|
||||||
static int nb_streamid_map = 0;
|
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);
|
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*/
|
/* copy chapters from the first input file that has them*/
|
||||||
|
if (!nb_chapter_maps)
|
||||||
for (i = 0; i < nb_input_files; i++) {
|
for (i = 0; i < nb_input_files; i++) {
|
||||||
if (!input_files[i]->nb_chapters)
|
if (!input_files[i]->nb_chapters)
|
||||||
continue;
|
continue;
|
||||||
@ -2962,6 +2991,21 @@ static void opt_map_meta_data(const char *arg)
|
|||||||
metadata_chapters_autocopy = 0;
|
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)
|
static void opt_input_ts_scale(const char *arg)
|
||||||
{
|
{
|
||||||
unsigned int stream;
|
unsigned int stream;
|
||||||
@ -4074,6 +4118,7 @@ static const OptionDef options[] = {
|
|||||||
{ "y", OPT_BOOL, {(void*)&file_overwrite}, "overwrite output files" },
|
{ "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", 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_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" },
|
{ "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" }, //
|
{ "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" },
|
{ "ss", OPT_FUNC2 | HAS_ARG, {(void*)opt_start_time}, "set the start time offset", "time_off" },
|
||||||
|
Loading…
Reference in New Issue
Block a user