fftools/cmdutils: add option syntax for loading arbitrary arguments from a file
Aligned with analogous feature for filter options in ffmpeg CLI.
This commit is contained in:
parent
8aed3911fc
commit
6d17991b7e
@ -20,6 +20,8 @@ version <next>:
|
|||||||
- fsync filter
|
- fsync filter
|
||||||
- Raw Captions with Time (RCWT) closed caption muxer
|
- Raw Captions with Time (RCWT) closed caption muxer
|
||||||
- ffmpeg CLI -bsf option may now be used for input as well as output
|
- 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:
|
version 6.1:
|
||||||
- libaribcaption decoder
|
- libaribcaption decoder
|
||||||
|
@ -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"
|
the option name with "no". For example using "-nofoo"
|
||||||
will set the boolean option with name "foo" to false.
|
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}
|
@anchor{Stream specifiers}
|
||||||
@section Stream specifiers
|
@section Stream specifiers
|
||||||
Some options are applied per-stream, e.g. bitrate or codec. Stream specifiers
|
Some options are applied per-stream, e.g. bitrate or codec. Stream specifiers
|
||||||
|
@ -37,6 +37,7 @@
|
|||||||
#include "libswresample/swresample.h"
|
#include "libswresample/swresample.h"
|
||||||
#include "libavutil/avassert.h"
|
#include "libavutil/avassert.h"
|
||||||
#include "libavutil/avstring.h"
|
#include "libavutil/avstring.h"
|
||||||
|
#include "libavutil/bprint.h"
|
||||||
#include "libavutil/channel_layout.h"
|
#include "libavutil/channel_layout.h"
|
||||||
#include "libavutil/display.h"
|
#include "libavutil/display.h"
|
||||||
#include "libavutil/getenv_utf8.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)
|
static const OptionDef *find_option(const OptionDef *po, const char *name)
|
||||||
{
|
{
|
||||||
|
if (*name == '/')
|
||||||
|
name++;
|
||||||
|
|
||||||
while (po->name) {
|
while (po->name) {
|
||||||
const char *end;
|
const char *end;
|
||||||
if (av_strstart(name, po->name, &end) && (!*end || *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*/
|
* a global var*/
|
||||||
void *dst = po->flags & OPT_FLAG_OFFSET ?
|
void *dst = po->flags & OPT_FLAG_OFFSET ?
|
||||||
(uint8_t *)optctx + po->u.off : po->u.dst_ptr;
|
(uint8_t *)optctx + po->u.off : po->u.dst_ptr;
|
||||||
|
char *arg_allocated = NULL;
|
||||||
|
|
||||||
SpecifierOptList *sol = NULL;
|
SpecifierOptList *sol = NULL;
|
||||||
double num;
|
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) {
|
if (po->flags & OPT_FLAG_SPEC) {
|
||||||
char *p = strchr(opt, ':');
|
char *p = strchr(opt, ':');
|
||||||
@ -250,32 +277,42 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt,
|
|||||||
sol = dst;
|
sol = dst;
|
||||||
ret = GROW_ARRAY(sol->opt, sol->nb_opt);
|
ret = GROW_ARRAY(sol->opt, sol->nb_opt);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto finish;
|
||||||
|
|
||||||
str = av_strdup(p ? p + 1 : "");
|
str = av_strdup(p ? p + 1 : "");
|
||||||
if (!str)
|
if (!str) {
|
||||||
return AVERROR(ENOMEM);
|
ret = AVERROR(ENOMEM);
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
sol->opt[sol->nb_opt - 1].specifier = str;
|
sol->opt[sol->nb_opt - 1].specifier = str;
|
||||||
dst = &sol->opt[sol->nb_opt - 1].u;
|
dst = &sol->opt[sol->nb_opt - 1].u;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (po->type == OPT_TYPE_STRING) {
|
if (po->type == OPT_TYPE_STRING) {
|
||||||
char *str;
|
char *str;
|
||||||
str = av_strdup(arg);
|
if (arg_allocated) {
|
||||||
|
str = arg_allocated;
|
||||||
|
arg_allocated = NULL;
|
||||||
|
} else
|
||||||
|
str = av_strdup(arg);
|
||||||
av_freep(dst);
|
av_freep(dst);
|
||||||
if (!str)
|
|
||||||
return AVERROR(ENOMEM);
|
if (!str) {
|
||||||
|
ret = AVERROR(ENOMEM);
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
|
||||||
*(char **)dst = str;
|
*(char **)dst = str;
|
||||||
} else if (po->type == OPT_TYPE_BOOL || po->type == OPT_TYPE_INT) {
|
} 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);
|
ret = parse_number(opt, arg, OPT_TYPE_INT64, INT_MIN, INT_MAX, &num);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto finish;
|
||||||
|
|
||||||
*(int *)dst = num;
|
*(int *)dst = num;
|
||||||
} else if (po->type == OPT_TYPE_INT64) {
|
} else if (po->type == OPT_TYPE_INT64) {
|
||||||
ret = parse_number(opt, arg, OPT_TYPE_INT64, INT64_MIN, INT64_MAX, &num);
|
ret = parse_number(opt, arg, OPT_TYPE_INT64, INT64_MIN, INT64_MAX, &num);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto finish;
|
||||||
|
|
||||||
*(int64_t *)dst = num;
|
*(int64_t *)dst = num;
|
||||||
} else if (po->type == OPT_TYPE_TIME) {
|
} 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) {
|
if (ret < 0) {
|
||||||
av_log(NULL, AV_LOG_ERROR, "Invalid duration for option %s: %s\n",
|
av_log(NULL, AV_LOG_ERROR, "Invalid duration for option %s: %s\n",
|
||||||
opt, arg);
|
opt, arg);
|
||||||
return ret;
|
goto finish;
|
||||||
}
|
}
|
||||||
} else if (po->type == OPT_TYPE_FLOAT) {
|
} else if (po->type == OPT_TYPE_FLOAT) {
|
||||||
ret = parse_number(opt, arg, OPT_TYPE_FLOAT, -INFINITY, INFINITY, &num);
|
ret = parse_number(opt, arg, OPT_TYPE_FLOAT, -INFINITY, INFINITY, &num);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto finish;
|
||||||
|
|
||||||
*(float *)dst = num;
|
*(float *)dst = num;
|
||||||
} else if (po->type == OPT_TYPE_DOUBLE) {
|
} else if (po->type == OPT_TYPE_DOUBLE) {
|
||||||
ret = parse_number(opt, arg, OPT_TYPE_DOUBLE, -INFINITY, INFINITY, &num);
|
ret = parse_number(opt, arg, OPT_TYPE_DOUBLE, -INFINITY, INFINITY, &num);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto finish;
|
||||||
|
|
||||||
*(double *)dst = num;
|
*(double *)dst = num;
|
||||||
} else {
|
} else {
|
||||||
@ -307,11 +344,13 @@ static int write_option(void *optctx, const OptionDef *po, const char *opt,
|
|||||||
av_log(NULL, AV_LOG_ERROR,
|
av_log(NULL, AV_LOG_ERROR,
|
||||||
"Failed to set value '%s' for option '%s': %s\n",
|
"Failed to set value '%s' for option '%s': %s\n",
|
||||||
arg, opt, av_err2str(ret));
|
arg, opt, av_err2str(ret));
|
||||||
return ret;
|
goto finish;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (po->flags & OPT_EXIT)
|
if (po->flags & OPT_EXIT) {
|
||||||
return AVERROR_EXIT;
|
ret = AVERROR_EXIT;
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
|
||||||
if (sol) {
|
if (sol) {
|
||||||
sol->type = po->type;
|
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;
|
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,
|
int parse_option(void *optctx, const char *opt, const char *arg,
|
||||||
@ -1088,3 +1129,29 @@ double get_rotation(const int32_t *displaymatrix)
|
|||||||
|
|
||||||
return theta;
|
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;
|
||||||
|
}
|
||||||
|
@ -470,4 +470,7 @@ void *allocate_array_elem(void *array, size_t elem_size, int *nb_elems);
|
|||||||
|
|
||||||
double get_rotation(const int32_t *displaymatrix);
|
double get_rotation(const int32_t *displaymatrix);
|
||||||
|
|
||||||
|
/* read file contents into a string */
|
||||||
|
char *file_read(const char *filename);
|
||||||
|
|
||||||
#endif /* FFTOOLS_CMDUTILS_H */
|
#endif /* FFTOOLS_CMDUTILS_H */
|
||||||
|
@ -652,7 +652,6 @@ void remove_avoptions(AVDictionary **a, AVDictionary *b);
|
|||||||
int check_avoptions(AVDictionary *m);
|
int check_avoptions(AVDictionary *m);
|
||||||
|
|
||||||
int assert_file_overwrite(const char *filename);
|
int assert_file_overwrite(const char *filename);
|
||||||
char *file_read(const char *filename);
|
|
||||||
AVDictionary *strip_specifiers(const AVDictionary *dict);
|
AVDictionary *strip_specifiers(const AVDictionary *dict);
|
||||||
int find_codec(void *logctx, const char *name,
|
int find_codec(void *logctx, const char *name,
|
||||||
enum AVMediaType type, int encoder, const AVCodec **codec);
|
enum AVMediaType type, int encoder, const AVCodec **codec);
|
||||||
|
@ -765,32 +765,6 @@ int assert_file_overwrite(const char *filename)
|
|||||||
return 0;
|
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". */
|
/* arg format is "output-stream-index:streamid-value". */
|
||||||
static int opt_streamid(void *optctx, const char *opt, const char *arg)
|
static int opt_streamid(void *optctx, const char *opt, const char *arg)
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user