diff options
author | Anton Khirnov <anton@khirnov.net> | 2022-03-15 09:16:18 +0100 |
---|---|---|
committer | Anton Khirnov <anton@khirnov.net> | 2024-01-20 10:23:24 +0100 |
commit | 6d17991b7e1bf1a5d104c8a6261709f7e6640d97 (patch) | |
tree | 26e527bec578435c492a9d944642cb01876a148d | |
parent | 8aed3911fc454e79697e183660bf30d31334a64b (diff) | |
download | ffmpeg-6d17991b7e1bf1a5d104c8a6261709f7e6640d97.tar.gz |
fftools/cmdutils: add option syntax for loading arbitrary arguments from a file
Aligned with analogous feature for filter options in ffmpeg CLI.
-rw-r--r-- | Changelog | 2 | ||||
-rw-r--r-- | doc/fftools-common-opts.texi | 9 | ||||
-rw-r--r-- | fftools/cmdutils.c | 99 | ||||
-rw-r--r-- | fftools/cmdutils.h | 3 | ||||
-rw-r--r-- | fftools/ffmpeg.h | 1 | ||||
-rw-r--r-- | fftools/ffmpeg_opt.c | 26 |
6 files changed, 97 insertions, 43 deletions
@@ -20,6 +20,8 @@ version <next>: - fsync filter - Raw Captions with Time (RCWT) closed caption muxer - ffmpeg CLI -bsf option may now be used for input as well as output +- ffmpeg CLI options may now be used as -/opt <path>, which is equivalent + to -opt <contents of file <path>> version 6.1: - libaribcaption decoder diff --git a/doc/fftools-common-opts.texi b/doc/fftools-common-opts.texi index f459bfdc1d..1974d79a4c 100644 --- a/doc/fftools-common-opts.texi +++ b/doc/fftools-common-opts.texi @@ -13,6 +13,15 @@ corresponding value to true. They can be set to false by prefixing the option name with "no". For example using "-nofoo" will set the boolean option with name "foo" to false. +Options that take arguments support a special syntax where the argument given on +the command line is interpreted as a path to the file from which the actual +argument value is loaded. To use this feature, add a forward slash '/' +immediately before the option name (after the leading dash). E.g. +@example +ffmpeg -i INPUT -/filter:v filter.script OUTPUT +@end example +will load a filtergraph description from the file named @file{filter.script}. + @anchor{Stream specifiers} @section Stream specifiers Some options are applied per-stream, e.g. bitrate or codec. Stream specifiers diff --git a/fftools/cmdutils.c b/fftools/cmdutils.c index 44228ea637..d7d5ddee45 100644 --- a/fftools/cmdutils.c +++ b/fftools/cmdutils.c @@ -37,6 +37,7 @@ #include "libswresample/swresample.h" #include "libavutil/avassert.h" #include "libavutil/avstring.h" +#include "libavutil/bprint.h" #include "libavutil/channel_layout.h" #include "libavutil/display.h" #include "libavutil/getenv_utf8.h" @@ -150,6 +151,9 @@ void show_help_children(const AVClass *class, int flags) static const OptionDef *find_option(const OptionDef *po, const char *name) { + if (*name == '/') + name++; + while (po->name) { const char *end; if (av_strstart(name, po->name, &end) && (!*end || *end == ':')) @@ -239,9 +243,32 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt, * a global var*/ void *dst = po->flags & OPT_FLAG_OFFSET ? (uint8_t *)optctx + po->u.off : po->u.dst_ptr; + char *arg_allocated = NULL; + SpecifierOptList *sol = NULL; double num; - int ret; + int ret = 0; + + if (*opt == '/') { + opt++; + + if (po->type == OPT_TYPE_BOOL) { + av_log(NULL, AV_LOG_FATAL, + "Requested to load an argument from file for a bool option '%s'\n", + po->name); + return AVERROR(EINVAL); + } + + arg_allocated = file_read(arg); + if (!arg_allocated) { + av_log(NULL, AV_LOG_FATAL, + "Error reading the value for option '%s' from file: %s\n", + opt, arg); + return AVERROR(EINVAL); + } + + arg = arg_allocated; + } if (po->flags & OPT_FLAG_SPEC) { char *p = strchr(opt, ':'); @@ -250,32 +277,42 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt, sol = dst; ret = GROW_ARRAY(sol->opt, sol->nb_opt); if (ret < 0) - return ret; + goto finish; str = av_strdup(p ? p + 1 : ""); - if (!str) - return AVERROR(ENOMEM); + if (!str) { + ret = AVERROR(ENOMEM); + goto finish; + } sol->opt[sol->nb_opt - 1].specifier = str; dst = &sol->opt[sol->nb_opt - 1].u; } if (po->type == OPT_TYPE_STRING) { char *str; - str = av_strdup(arg); + if (arg_allocated) { + str = arg_allocated; + arg_allocated = NULL; + } else + str = av_strdup(arg); av_freep(dst); - if (!str) - return AVERROR(ENOMEM); + + if (!str) { + ret = AVERROR(ENOMEM); + goto finish; + } + *(char **)dst = str; } else if (po->type == OPT_TYPE_BOOL || po->type == OPT_TYPE_INT) { ret = parse_number(opt, arg, OPT_TYPE_INT64, INT_MIN, INT_MAX, &num); if (ret < 0) - return ret; + goto finish; *(int *)dst = num; } else if (po->type == OPT_TYPE_INT64) { ret = parse_number(opt, arg, OPT_TYPE_INT64, INT64_MIN, INT64_MAX, &num); if (ret < 0) - return ret; + goto finish; *(int64_t *)dst = num; } else if (po->type == OPT_TYPE_TIME) { @@ -283,18 +320,18 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt, if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Invalid duration for option %s: %s\n", opt, arg); - return ret; + goto finish; } } else if (po->type == OPT_TYPE_FLOAT) { ret = parse_number(opt, arg, OPT_TYPE_FLOAT, -INFINITY, INFINITY, &num); if (ret < 0) - return ret; + goto finish; *(float *)dst = num; } else if (po->type == OPT_TYPE_DOUBLE) { ret = parse_number(opt, arg, OPT_TYPE_DOUBLE, -INFINITY, INFINITY, &num); if (ret < 0) - return ret; + goto finish; *(double *)dst = num; } else { @@ -307,11 +344,13 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt, av_log(NULL, AV_LOG_ERROR, "Failed to set value '%s' for option '%s': %s\n", arg, opt, av_err2str(ret)); - return ret; + goto finish; } } - if (po->flags & OPT_EXIT) - return AVERROR_EXIT; + if (po->flags & OPT_EXIT) { + ret = AVERROR_EXIT; + goto finish; + } if (sol) { sol->type = po->type; @@ -319,7 +358,9 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt, find_option(defs, po->u1.name_canon) : po; } - return 0; +finish: + av_freep(&arg_allocated); + return ret; } int parse_option(void *optctx, const char *opt, const char *arg, @@ -1088,3 +1129,29 @@ double get_rotation(const int32_t *displaymatrix) return theta; } + +/* read file contents into a string */ +char *file_read(const char *filename) +{ + AVIOContext *pb = NULL; + int ret = avio_open(&pb, filename, AVIO_FLAG_READ); + AVBPrint bprint; + char *str; + + if (ret < 0) { + av_log(NULL, AV_LOG_ERROR, "Error opening file %s.\n", filename); + return NULL; + } + + av_bprint_init(&bprint, 0, AV_BPRINT_SIZE_UNLIMITED); + ret = avio_read_to_bprint(pb, &bprint, SIZE_MAX); + avio_closep(&pb); + if (ret < 0) { + av_bprint_finalize(&bprint, NULL); + return NULL; + } + ret = av_bprint_finalize(&bprint, &str); + if (ret < 0) + return NULL; + return str; +} diff --git a/fftools/cmdutils.h b/fftools/cmdutils.h index 53227abb47..8fa5ad4fc7 100644 --- a/fftools/cmdutils.h +++ b/fftools/cmdutils.h @@ -470,4 +470,7 @@ void *allocate_array_elem(void *array, size_t elem_size, int *nb_elems); double get_rotation(const int32_t *displaymatrix); +/* read file contents into a string */ +char *file_read(const char *filename); + #endif /* FFTOOLS_CMDUTILS_H */ diff --git a/fftools/ffmpeg.h b/fftools/ffmpeg.h index 6137ac991e..cdde3c2c03 100644 --- a/fftools/ffmpeg.h +++ b/fftools/ffmpeg.h @@ -652,7 +652,6 @@ void remove_avoptions(AVDictionary **a, AVDictionary *b); int check_avoptions(AVDictionary *m); int assert_file_overwrite(const char *filename); -char *file_read(const char *filename); AVDictionary *strip_specifiers(const AVDictionary *dict); int find_codec(void *logctx, const char *name, enum AVMediaType type, int encoder, const AVCodec **codec); diff --git a/fftools/ffmpeg_opt.c b/fftools/ffmpeg_opt.c index 563f443bd8..ffb2c42421 100644 --- a/fftools/ffmpeg_opt.c +++ b/fftools/ffmpeg_opt.c @@ -765,32 +765,6 @@ int assert_file_overwrite(const char *filename) return 0; } -/* read file contents into a string */ -char *file_read(const char *filename) -{ - AVIOContext *pb = NULL; - int ret = avio_open(&pb, filename, AVIO_FLAG_READ); - AVBPrint bprint; - char *str; - - if (ret < 0) { - av_log(NULL, AV_LOG_ERROR, "Error opening file %s.\n", filename); - return NULL; - } - - av_bprint_init(&bprint, 0, AV_BPRINT_SIZE_UNLIMITED); - ret = avio_read_to_bprint(pb, &bprint, SIZE_MAX); - avio_closep(&pb); - if (ret < 0) { - av_bprint_finalize(&bprint, NULL); - return NULL; - } - ret = av_bprint_finalize(&bprint, &str); - if (ret < 0) - return NULL; - return str; -} - /* arg format is "output-stream-index:streamid-value". */ static int opt_streamid(void *optctx, const char *opt, const char *arg) { |