Commit | Line | Data |
---|---|---|
01310af2 FB |
1 | /* |
2 | * Various utilities for command line tools | |
3 | * Copyright (c) 2000-2003 Fabrice Bellard | |
4 | * | |
b78e7197 DB |
5 | * This file is part of FFmpeg. |
6 | * | |
7 | * FFmpeg is free software; you can redistribute it and/or | |
01310af2 FB |
8 | * modify it under the terms of the GNU Lesser General Public |
9 | * License as published by the Free Software Foundation; either | |
b78e7197 | 10 | * version 2.1 of the License, or (at your option) any later version. |
01310af2 | 11 | * |
b78e7197 | 12 | * FFmpeg is distributed in the hope that it will be useful, |
01310af2 FB |
13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
15 | * Lesser General Public License for more details. | |
16 | * | |
17 | * You should have received a copy of the GNU Lesser General Public | |
b78e7197 | 18 | * License along with FFmpeg; if not, write to the Free Software |
5509bffa | 19 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
01310af2 | 20 | */ |
364a9607 | 21 | |
0f4e8165 RB |
22 | #include <string.h> |
23 | #include <stdlib.h> | |
24 | #include <errno.h> | |
7c84b8bc | 25 | #include <math.h> |
0f4e8165 | 26 | |
2cd39dbf DP |
27 | /* Include only the enabled headers since some compilers (namely, Sun |
28 | Studio) will not omit unused inline functions and create undefined | |
29 | references to libraries that are not being built. */ | |
30 | ||
63d026b1 | 31 | #include "config.h" |
245976da DB |
32 | #include "libavformat/avformat.h" |
33 | #include "libavfilter/avfilter.h" | |
34 | #include "libavdevice/avdevice.h" | |
db6d50c7 | 35 | #include "libswscale/swscale.h" |
1981deaf | 36 | #include "libpostproc/postprocess.h" |
245976da | 37 | #include "libavutil/avstring.h" |
9cb5c760 | 38 | #include "libavutil/pixdesc.h" |
25e25617 | 39 | #include "libavutil/eval.h" |
85663ef3 | 40 | #include "libavcodec/opt.h" |
01310af2 | 41 | #include "cmdutils.h" |
86074ed1 | 42 | #include "version.h" |
da2dc39e | 43 | #if CONFIG_NETWORK |
245976da | 44 | #include "libavformat/network.h" |
da2dc39e | 45 | #endif |
ffcc6e24 MR |
46 | #if HAVE_SYS_RESOURCE_H |
47 | #include <sys/resource.h> | |
48 | #endif | |
01310af2 | 49 | |
85663ef3 MN |
50 | const char **opt_names; |
51 | static int opt_name_count; | |
72415b2a | 52 | AVCodecContext *avcodec_opts[AVMEDIA_TYPE_NB]; |
85663ef3 MN |
53 | AVFormatContext *avformat_opts; |
54 | struct SwsContext *sws_opts; | |
086ab001 | 55 | |
7a78bc85 | 56 | const int this_year = 2010; |
ef4c0bb1 | 57 | |
086ab001 MN |
58 | double parse_number_or_die(const char *context, const char *numstr, int type, double min, double max) |
59 | { | |
60 | char *tail; | |
61 | const char *error; | |
25e25617 | 62 | double d = av_strtod(numstr, &tail); |
086ab001 MN |
63 | if (*tail) |
64 | error= "Expected number for %s but found: %s\n"; | |
65 | else if (d < min || d > max) | |
66 | error= "The value for %s was %s which is not within %f - %f\n"; | |
67 | else if(type == OPT_INT64 && (int64_t)d != d) | |
68 | error= "Expected int64 for %s but found %s\n"; | |
69 | else | |
70 | return d; | |
71 | fprintf(stderr, error, context, numstr, min, max); | |
72 | exit(1); | |
73 | } | |
74 | ||
7542157d SS |
75 | int64_t parse_time_or_die(const char *context, const char *timestr, int is_duration) |
76 | { | |
77 | int64_t us = parse_date(timestr, is_duration); | |
78 | if (us == INT64_MIN) { | |
79 | fprintf(stderr, "Invalid %s specification for %s: %s\n", | |
80 | is_duration ? "duration" : "date", context, timestr); | |
81 | exit(1); | |
82 | } | |
83 | return us; | |
84 | } | |
85 | ||
02d504a7 | 86 | void show_help_options(const OptionDef *options, const char *msg, int mask, int value) |
01310af2 FB |
87 | { |
88 | const OptionDef *po; | |
02d504a7 | 89 | int first; |
01310af2 | 90 | |
02d504a7 FB |
91 | first = 1; |
92 | for(po = options; po->name != NULL; po++) { | |
93 | char buf[64]; | |
94 | if ((po->flags & mask) == value) { | |
95 | if (first) { | |
96 | printf("%s", msg); | |
97 | first = 0; | |
98 | } | |
f7d78f36 | 99 | av_strlcpy(buf, po->name, sizeof(buf)); |
02d504a7 | 100 | if (po->flags & HAS_ARG) { |
f7d78f36 MR |
101 | av_strlcat(buf, " ", sizeof(buf)); |
102 | av_strlcat(buf, po->argname, sizeof(buf)); | |
01310af2 | 103 | } |
02d504a7 | 104 | printf("-%-17s %s\n", buf, po->help); |
01310af2 FB |
105 | } |
106 | } | |
107 | } | |
108 | ||
fccfc475 | 109 | static const OptionDef* find_option(const OptionDef *po, const char *name){ |
8bbf6db9 MN |
110 | while (po->name != NULL) { |
111 | if (!strcmp(name, po->name)) | |
112 | break; | |
113 | po++; | |
114 | } | |
115 | return po; | |
116 | } | |
117 | ||
60a9966e SS |
118 | void parse_options(int argc, char **argv, const OptionDef *options, |
119 | void (* parse_arg_function)(const char*)) | |
01310af2 FB |
120 | { |
121 | const char *opt, *arg; | |
b0d7bc1e | 122 | int optindex, handleoptions=1; |
01310af2 FB |
123 | const OptionDef *po; |
124 | ||
125 | /* parse options */ | |
126 | optindex = 1; | |
127 | while (optindex < argc) { | |
128 | opt = argv[optindex++]; | |
115329f1 | 129 | |
84bf226b | 130 | if (handleoptions && opt[0] == '-' && opt[1] != '\0') { |
b1d6e5e8 | 131 | int bool_val = 1; |
3749076c SS |
132 | if (opt[1] == '-' && opt[2] == '\0') { |
133 | handleoptions = 0; | |
134 | continue; | |
135 | } | |
c3c78324 SS |
136 | opt++; |
137 | po= find_option(options, opt); | |
138 | if (!po->name && opt[0] == 'n' && opt[1] == 'o') { | |
b1d6e5e8 | 139 | /* handle 'no' bool option */ |
c3c78324 | 140 | po = find_option(options, opt + 2); |
b1d6e5e8 BF |
141 | if (!(po->name && (po->flags & OPT_BOOL))) |
142 | goto unknown_opt; | |
143 | bool_val = 0; | |
144 | } | |
8bbf6db9 MN |
145 | if (!po->name) |
146 | po= find_option(options, "default"); | |
01310af2 | 147 | if (!po->name) { |
8bbf6db9 | 148 | unknown_opt: |
01310af2 FB |
149 | fprintf(stderr, "%s: unrecognized option '%s'\n", argv[0], opt); |
150 | exit(1); | |
151 | } | |
152 | arg = NULL; | |
153 | if (po->flags & HAS_ARG) { | |
154 | arg = argv[optindex++]; | |
155 | if (!arg) { | |
156 | fprintf(stderr, "%s: missing argument for option '%s'\n", argv[0], opt); | |
157 | exit(1); | |
158 | } | |
159 | } | |
160 | if (po->flags & OPT_STRING) { | |
161 | char *str; | |
02d504a7 | 162 | str = av_strdup(arg); |
01310af2 FB |
163 | *po->u.str_arg = str; |
164 | } else if (po->flags & OPT_BOOL) { | |
b1d6e5e8 | 165 | *po->u.int_arg = bool_val; |
26d4f26b | 166 | } else if (po->flags & OPT_INT) { |
c3c78324 | 167 | *po->u.int_arg = parse_number_or_die(opt, arg, OPT_INT64, INT_MIN, INT_MAX); |
ffdf9a1f | 168 | } else if (po->flags & OPT_INT64) { |
c3c78324 | 169 | *po->u.int64_arg = parse_number_or_die(opt, arg, OPT_INT64, INT64_MIN, INT64_MAX); |
1f631450 | 170 | } else if (po->flags & OPT_FLOAT) { |
c3c78324 | 171 | *po->u.float_arg = parse_number_or_die(opt, arg, OPT_FLOAT, -1.0/0.0, 1.0/0.0); |
8bbf6db9 | 172 | } else if (po->flags & OPT_FUNC2) { |
9e5381a2 | 173 | if (po->u.func2_arg(opt, arg) < 0) { |
330d86f5 | 174 | fprintf(stderr, "%s: failed to set value '%s' for option '%s'\n", argv[0], arg, opt); |
9e5381a2 SS |
175 | exit(1); |
176 | } | |
01310af2 | 177 | } else { |
bb270c08 | 178 | po->u.func_arg(arg); |
01310af2 | 179 | } |
a0b3bcd9 MN |
180 | if(po->flags & OPT_EXIT) |
181 | exit(0); | |
01310af2 | 182 | } else { |
60a9966e SS |
183 | if (parse_arg_function) |
184 | parse_arg_function(opt); | |
01310af2 FB |
185 | } |
186 | } | |
187 | } | |
188 | ||
85663ef3 MN |
189 | int opt_default(const char *opt, const char *arg){ |
190 | int type; | |
5c3383e5 | 191 | int ret= 0; |
85663ef3 MN |
192 | const AVOption *o= NULL; |
193 | int opt_types[]={AV_OPT_FLAG_VIDEO_PARAM, AV_OPT_FLAG_AUDIO_PARAM, 0, AV_OPT_FLAG_SUBTITLE_PARAM, 0}; | |
194 | ||
72415b2a | 195 | for(type=0; type<AVMEDIA_TYPE_NB && ret>= 0; type++){ |
636f1c4c | 196 | const AVOption *o2 = av_find_opt(avcodec_opts[0], opt, NULL, opt_types[type], opt_types[type]); |
85663ef3 | 197 | if(o2) |
636f1c4c | 198 | ret = av_set_string3(avcodec_opts[type], opt, arg, 1, &o); |
85663ef3 MN |
199 | } |
200 | if(!o) | |
5c3383e5 | 201 | ret = av_set_string3(avformat_opts, opt, arg, 1, &o); |
da033b05 | 202 | if(!o && sws_opts) |
5c3383e5 | 203 | ret = av_set_string3(sws_opts, opt, arg, 1, &o); |
85663ef3 MN |
204 | if(!o){ |
205 | if(opt[0] == 'a') | |
72415b2a | 206 | ret = av_set_string3(avcodec_opts[AVMEDIA_TYPE_AUDIO], opt+1, arg, 1, &o); |
85663ef3 | 207 | else if(opt[0] == 'v') |
72415b2a | 208 | ret = av_set_string3(avcodec_opts[AVMEDIA_TYPE_VIDEO], opt+1, arg, 1, &o); |
85663ef3 | 209 | else if(opt[0] == 's') |
72415b2a | 210 | ret = av_set_string3(avcodec_opts[AVMEDIA_TYPE_SUBTITLE], opt+1, arg, 1, &o); |
5c3383e5 SS |
211 | } |
212 | if (o && ret < 0) { | |
213 | fprintf(stderr, "Invalid value '%s' for option '%s'\n", arg, opt); | |
214 | exit(1); | |
85663ef3 | 215 | } |
33bc7947 SS |
216 | if (!o) { |
217 | fprintf(stderr, "Unrecognized option '%s'\n", opt); | |
218 | exit(1); | |
219 | } | |
85663ef3 | 220 | |
636f1c4c | 221 | // av_log(NULL, AV_LOG_ERROR, "%s:%s: %f 0x%0X\n", opt, arg, av_get_double(avcodec_opts, opt, NULL), (int)av_get_int(avcodec_opts, opt, NULL)); |
85663ef3 | 222 | |
636f1c4c | 223 | //FIXME we should always use avcodec_opts, ... for storing options so there will not be any need to keep track of what i set over this |
85663ef3 MN |
224 | opt_names= av_realloc(opt_names, sizeof(void*)*(opt_name_count+1)); |
225 | opt_names[opt_name_count++]= o->name; | |
226 | ||
636f1c4c | 227 | if(avcodec_opts[0]->debug || avformat_opts->debug) |
85663ef3 MN |
228 | av_log_set_level(AV_LOG_DEBUG); |
229 | return 0; | |
230 | } | |
231 | ||
4c97a6fa SS |
232 | int opt_loglevel(const char *opt, const char *arg) |
233 | { | |
da4c2dab | 234 | const struct { const char *name; int level; } log_levels[] = { |
4c97a6fa SS |
235 | { "quiet" , AV_LOG_QUIET }, |
236 | { "panic" , AV_LOG_PANIC }, | |
237 | { "fatal" , AV_LOG_FATAL }, | |
238 | { "error" , AV_LOG_ERROR }, | |
239 | { "warning", AV_LOG_WARNING }, | |
240 | { "info" , AV_LOG_INFO }, | |
241 | { "verbose", AV_LOG_VERBOSE }, | |
242 | { "debug" , AV_LOG_DEBUG }, | |
243 | }; | |
244 | char *tail; | |
245 | int level; | |
246 | int i; | |
247 | ||
248 | for (i = 0; i < FF_ARRAY_ELEMS(log_levels); i++) { | |
249 | if (!strcmp(log_levels[i].name, arg)) { | |
250 | av_log_set_level(log_levels[i].level); | |
251 | return 0; | |
252 | } | |
253 | } | |
254 | ||
255 | level = strtol(arg, &tail, 10); | |
256 | if (*tail) { | |
257 | fprintf(stderr, "Invalid loglevel \"%s\". " | |
258 | "Possible levels are numbers or:\n", arg); | |
259 | for (i = 0; i < FF_ARRAY_ELEMS(log_levels); i++) | |
260 | fprintf(stderr, "\"%s\"\n", log_levels[i].name); | |
261 | exit(1); | |
262 | } | |
263 | av_log_set_level(level); | |
264 | return 0; | |
265 | } | |
266 | ||
ffcc6e24 MR |
267 | int opt_timelimit(const char *opt, const char *arg) |
268 | { | |
0104b608 | 269 | #if HAVE_SETRLIMIT |
ffcc6e24 MR |
270 | int lim = parse_number_or_die(opt, arg, OPT_INT64, 0, INT_MAX); |
271 | struct rlimit rl = { lim, lim + 1 }; | |
272 | if (setrlimit(RLIMIT_CPU, &rl)) | |
273 | perror("setrlimit"); | |
274 | #else | |
275 | fprintf(stderr, "Warning: -%s not implemented on this OS\n", opt); | |
276 | #endif | |
277 | return 0; | |
278 | } | |
279 | ||
85663ef3 MN |
280 | void set_context_opts(void *ctx, void *opts_ctx, int flags) |
281 | { | |
282 | int i; | |
283 | for(i=0; i<opt_name_count; i++){ | |
284 | char buf[256]; | |
285 | const AVOption *opt; | |
286 | const char *str= av_get_string(opts_ctx, opt_names[i], &opt, buf, sizeof(buf)); | |
287 | /* if an option with name opt_names[i] is present in opts_ctx then str is non-NULL */ | |
288 | if(str && ((opt->flags & flags) == flags)) | |
f16dd7e6 | 289 | av_set_string3(ctx, opt_names[i], str, 1, NULL); |
85663ef3 MN |
290 | } |
291 | } | |
292 | ||
01310af2 FB |
293 | void print_error(const char *filename, int err) |
294 | { | |
7a5bdd79 | 295 | char errbuf[128]; |
735ef67b | 296 | const char *errbuf_ptr = errbuf; |
7a5bdd79 | 297 | |
9e94bd3e SS |
298 | if (av_strerror(err, errbuf, sizeof(errbuf)) < 0) |
299 | errbuf_ptr = strerror(AVUNERROR(err)); | |
300 | fprintf(stderr, "%s: %s\n", filename, errbuf_ptr); | |
01310af2 | 301 | } |
f35917b2 | 302 | |
5116571d MR |
303 | #define PRINT_LIB_VERSION(outstream,libname,LIBNAME,indent) \ |
304 | if (CONFIG_##LIBNAME) { \ | |
695f7de6 | 305 | unsigned int version = libname##_version(); \ |
5116571d MR |
306 | fprintf(outstream, "%slib%-10s %2d.%2d.%2d / %2d.%2d.%2d\n", \ |
307 | indent? " " : "", #libname, \ | |
308 | LIB##LIBNAME##_VERSION_MAJOR, \ | |
309 | LIB##LIBNAME##_VERSION_MINOR, \ | |
310 | LIB##LIBNAME##_VERSION_MICRO, \ | |
311 | version >> 16, version >> 8 & 0xff, version & 0xff); \ | |
58fe804a | 312 | } |
9a109272 | 313 | |
88b77ef1 | 314 | static void print_all_lib_versions(FILE* outstream, int indent) |
9a109272 | 315 | { |
e9df66a7 SS |
316 | PRINT_LIB_VERSION(outstream, avutil, AVUTIL, indent); |
317 | PRINT_LIB_VERSION(outstream, avcodec, AVCODEC, indent); | |
9a109272 SS |
318 | PRINT_LIB_VERSION(outstream, avformat, AVFORMAT, indent); |
319 | PRINT_LIB_VERSION(outstream, avdevice, AVDEVICE, indent); | |
9a109272 | 320 | PRINT_LIB_VERSION(outstream, avfilter, AVFILTER, indent); |
e9df66a7 | 321 | PRINT_LIB_VERSION(outstream, swscale, SWSCALE, indent); |
1981deaf | 322 | PRINT_LIB_VERSION(outstream, postproc, POSTPROC, indent); |
9a109272 SS |
323 | } |
324 | ||
9120e2cd MR |
325 | static void maybe_print_config(const char *lib, const char *cfg) |
326 | { | |
327 | static int warned_cfg; | |
328 | ||
329 | if (strcmp(FFMPEG_CONFIGURATION, cfg)) { | |
330 | if (!warned_cfg) { | |
331 | fprintf(stderr, " WARNING: library configuration mismatch\n"); | |
332 | warned_cfg = 1; | |
333 | } | |
334 | fprintf(stderr, " %-11s configuration: %s\n", lib, cfg); | |
335 | } | |
336 | } | |
337 | ||
58fe804a MR |
338 | #define PRINT_LIB_CONFIG(lib, tag, cfg) do { \ |
339 | if (CONFIG_##lib) \ | |
340 | maybe_print_config(tag, cfg); \ | |
341 | } while (0) | |
342 | ||
ea9c581f | 343 | void show_banner(void) |
86074ed1 | 344 | { |
13887093 | 345 | fprintf(stderr, "%s version " FFMPEG_VERSION ", Copyright (c) %d-%d the FFmpeg developers\n", |
ef4c0bb1 | 346 | program_name, program_birth_year, this_year); |
a3d7c197 DB |
347 | fprintf(stderr, " built on %s %s with %s %s\n", |
348 | __DATE__, __TIME__, CC_TYPE, CC_VERSION); | |
5c1f57ff | 349 | fprintf(stderr, " configuration: " FFMPEG_CONFIGURATION "\n"); |
58fe804a MR |
350 | PRINT_LIB_CONFIG(AVUTIL, "libavutil", avutil_configuration()); |
351 | PRINT_LIB_CONFIG(AVCODEC, "libavcodec", avcodec_configuration()); | |
352 | PRINT_LIB_CONFIG(AVFORMAT, "libavformat", avformat_configuration()); | |
353 | PRINT_LIB_CONFIG(AVDEVICE, "libavdevice", avdevice_configuration()); | |
354 | PRINT_LIB_CONFIG(AVFILTER, "libavfilter", avfilter_configuration()); | |
355 | PRINT_LIB_CONFIG(SWSCALE, "libswscale", swscale_configuration()); | |
356 | PRINT_LIB_CONFIG(POSTPROC, "libpostproc", postproc_configuration()); | |
5c1f57ff | 357 | print_all_lib_versions(stderr, 1); |
86074ed1 SS |
358 | } |
359 | ||
64555bd9 | 360 | void show_version(void) { |
86074ed1 | 361 | printf("%s " FFMPEG_VERSION "\n", program_name); |
9a109272 | 362 | print_all_lib_versions(stdout, 0); |
86074ed1 SS |
363 | } |
364 | ||
f35917b2 SS |
365 | void show_license(void) |
366 | { | |
7ead693b | 367 | printf( |
3bf28f9d | 368 | #if CONFIG_NONFREE |
304ba23a SS |
369 | "This version of %s has nonfree parts compiled in.\n" |
370 | "Therefore it is not legally redistributable.\n", | |
371 | program_name | |
9cad0e4e DB |
372 | #elif CONFIG_GPLV3 |
373 | "%s is free software; you can redistribute it and/or modify\n" | |
374 | "it under the terms of the GNU General Public License as published by\n" | |
375 | "the Free Software Foundation; either version 3 of the License, or\n" | |
376 | "(at your option) any later version.\n" | |
377 | "\n" | |
378 | "%s is distributed in the hope that it will be useful,\n" | |
379 | "but WITHOUT ANY WARRANTY; without even the implied warranty of\n" | |
380 | "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n" | |
381 | "GNU General Public License for more details.\n" | |
382 | "\n" | |
383 | "You should have received a copy of the GNU General Public License\n" | |
384 | "along with %s. If not, see <http://www.gnu.org/licenses/>.\n", | |
385 | program_name, program_name, program_name | |
7ead693b | 386 | #elif CONFIG_GPL |
304ba23a | 387 | "%s is free software; you can redistribute it and/or modify\n" |
f35917b2 SS |
388 | "it under the terms of the GNU General Public License as published by\n" |
389 | "the Free Software Foundation; either version 2 of the License, or\n" | |
390 | "(at your option) any later version.\n" | |
391 | "\n" | |
304ba23a | 392 | "%s is distributed in the hope that it will be useful,\n" |
f35917b2 SS |
393 | "but WITHOUT ANY WARRANTY; without even the implied warranty of\n" |
394 | "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n" | |
395 | "GNU General Public License for more details.\n" | |
396 | "\n" | |
397 | "You should have received a copy of the GNU General Public License\n" | |
304ba23a SS |
398 | "along with %s; if not, write to the Free Software\n" |
399 | "Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n", | |
400 | program_name, program_name, program_name | |
9cad0e4e DB |
401 | #elif CONFIG_LGPLV3 |
402 | "%s is free software; you can redistribute it and/or modify\n" | |
403 | "it under the terms of the GNU Lesser General Public License as published by\n" | |
404 | "the Free Software Foundation; either version 3 of the License, or\n" | |
405 | "(at your option) any later version.\n" | |
406 | "\n" | |
407 | "%s is distributed in the hope that it will be useful,\n" | |
408 | "but WITHOUT ANY WARRANTY; without even the implied warranty of\n" | |
409 | "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n" | |
410 | "GNU Lesser General Public License for more details.\n" | |
411 | "\n" | |
412 | "You should have received a copy of the GNU Lesser General Public License\n" | |
413 | "along with %s. If not, see <http://www.gnu.org/licenses/>.\n", | |
414 | program_name, program_name, program_name | |
f35917b2 | 415 | #else |
304ba23a | 416 | "%s is free software; you can redistribute it and/or\n" |
f35917b2 SS |
417 | "modify it under the terms of the GNU Lesser General Public\n" |
418 | "License as published by the Free Software Foundation; either\n" | |
419 | "version 2.1 of the License, or (at your option) any later version.\n" | |
420 | "\n" | |
304ba23a | 421 | "%s is distributed in the hope that it will be useful,\n" |
f35917b2 SS |
422 | "but WITHOUT ANY WARRANTY; without even the implied warranty of\n" |
423 | "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n" | |
424 | "Lesser General Public License for more details.\n" | |
425 | "\n" | |
426 | "You should have received a copy of the GNU Lesser General Public\n" | |
304ba23a SS |
427 | "License along with %s; if not, write to the Free Software\n" |
428 | "Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n", | |
429 | program_name, program_name, program_name | |
f35917b2 | 430 | #endif |
3bf28f9d | 431 | ); |
f35917b2 | 432 | } |
ba9880c1 | 433 | |
c5dc6026 SS |
434 | void list_fmts(void (*get_fmt_string)(char *buf, int buf_size, int fmt), int nb_fmts) |
435 | { | |
436 | int i; | |
437 | char fmt_str[128]; | |
438 | for (i=-1; i < nb_fmts; i++) { | |
439 | get_fmt_string (fmt_str, sizeof(fmt_str), i); | |
440 | fprintf(stdout, "%s\n", fmt_str); | |
441 | } | |
442 | } | |
443 | ||
ba9880c1 SS |
444 | void show_formats(void) |
445 | { | |
446 | AVInputFormat *ifmt=NULL; | |
447 | AVOutputFormat *ofmt=NULL; | |
ba9880c1 SS |
448 | const char *last_name; |
449 | ||
5a8597a0 WZ |
450 | printf( |
451 | "File formats:\n" | |
452 | " D. = Demuxing supported\n" | |
453 | " .E = Muxing supported\n" | |
454 | " --\n"); | |
ba9880c1 SS |
455 | last_name= "000"; |
456 | for(;;){ | |
457 | int decode=0; | |
458 | int encode=0; | |
459 | const char *name=NULL; | |
460 | const char *long_name=NULL; | |
461 | ||
462 | while((ofmt= av_oformat_next(ofmt))) { | |
463 | if((name == NULL || strcmp(ofmt->name, name)<0) && | |
464 | strcmp(ofmt->name, last_name)>0){ | |
465 | name= ofmt->name; | |
466 | long_name= ofmt->long_name; | |
467 | encode=1; | |
468 | } | |
469 | } | |
470 | while((ifmt= av_iformat_next(ifmt))) { | |
471 | if((name == NULL || strcmp(ifmt->name, name)<0) && | |
472 | strcmp(ifmt->name, last_name)>0){ | |
473 | name= ifmt->name; | |
474 | long_name= ifmt->long_name; | |
475 | encode=0; | |
476 | } | |
477 | if(name && strcmp(ifmt->name, name)==0) | |
478 | decode=1; | |
479 | } | |
480 | if(name==NULL) | |
481 | break; | |
482 | last_name= name; | |
483 | ||
484 | printf( | |
485 | " %s%s %-15s %s\n", | |
486 | decode ? "D":" ", | |
487 | encode ? "E":" ", | |
488 | name, | |
489 | long_name ? long_name:" "); | |
490 | } | |
8447f0bd | 491 | } |
ba9880c1 | 492 | |
8447f0bd MN |
493 | void show_codecs(void) |
494 | { | |
495 | AVCodec *p=NULL, *p2; | |
496 | const char *last_name; | |
5a8597a0 WZ |
497 | printf( |
498 | "Codecs:\n" | |
499 | " D..... = Decoding supported\n" | |
500 | " .E.... = Encoding supported\n" | |
501 | " ..V... = Video codec\n" | |
502 | " ..A... = Audio codec\n" | |
503 | " ..S... = Subtitle codec\n" | |
504 | " ...S.. = Supports draw_horiz_band\n" | |
505 | " ....D. = Supports direct rendering method 1\n" | |
506 | " .....T = Supports weird frame truncation\n" | |
507 | " ------\n"); | |
ba9880c1 SS |
508 | last_name= "000"; |
509 | for(;;){ | |
510 | int decode=0; | |
511 | int encode=0; | |
512 | int cap=0; | |
513 | const char *type_str; | |
514 | ||
515 | p2=NULL; | |
516 | while((p= av_codec_next(p))) { | |
517 | if((p2==NULL || strcmp(p->name, p2->name)<0) && | |
518 | strcmp(p->name, last_name)>0){ | |
519 | p2= p; | |
520 | decode= encode= cap=0; | |
521 | } | |
522 | if(p2 && strcmp(p->name, p2->name)==0){ | |
523 | if(p->decode) decode=1; | |
524 | if(p->encode) encode=1; | |
525 | cap |= p->capabilities; | |
526 | } | |
527 | } | |
528 | if(p2==NULL) | |
529 | break; | |
530 | last_name= p2->name; | |
531 | ||
532 | switch(p2->type) { | |
72415b2a | 533 | case AVMEDIA_TYPE_VIDEO: |
ba9880c1 SS |
534 | type_str = "V"; |
535 | break; | |
72415b2a | 536 | case AVMEDIA_TYPE_AUDIO: |
ba9880c1 SS |
537 | type_str = "A"; |
538 | break; | |
72415b2a | 539 | case AVMEDIA_TYPE_SUBTITLE: |
ba9880c1 SS |
540 | type_str = "S"; |
541 | break; | |
542 | default: | |
543 | type_str = "?"; | |
544 | break; | |
545 | } | |
546 | printf( | |
547 | " %s%s%s%s%s%s %-15s %s", | |
548 | decode ? "D": (/*p2->decoder ? "d":*/" "), | |
549 | encode ? "E":" ", | |
550 | type_str, | |
551 | cap & CODEC_CAP_DRAW_HORIZ_BAND ? "S":" ", | |
552 | cap & CODEC_CAP_DR1 ? "D":" ", | |
553 | cap & CODEC_CAP_TRUNCATED ? "T":" ", | |
554 | p2->name, | |
555 | p2->long_name ? p2->long_name : ""); | |
556 | /* if(p2->decoder && decode==0) | |
557 | printf(" use %s for decoding", p2->decoder->name);*/ | |
558 | printf("\n"); | |
559 | } | |
560 | printf("\n"); | |
8447f0bd MN |
561 | printf( |
562 | "Note, the names of encoders and decoders do not always match, so there are\n" | |
563 | "several cases where the above table shows encoder only or decoder only entries\n" | |
564 | "even though both encoding and decoding are supported. For example, the h263\n" | |
565 | "decoder corresponds to the h263 and h263p encoders, for file formats it is even\n" | |
566 | "worse.\n"); | |
567 | } | |
568 | ||
569 | void show_bsfs(void) | |
570 | { | |
571 | AVBitStreamFilter *bsf=NULL; | |
ba9880c1 SS |
572 | |
573 | printf("Bitstream filters:\n"); | |
574 | while((bsf = av_bitstream_filter_next(bsf))) | |
2091b27b | 575 | printf("%s\n", bsf->name); |
ba9880c1 | 576 | printf("\n"); |
8447f0bd MN |
577 | } |
578 | ||
579 | void show_protocols(void) | |
580 | { | |
581 | URLProtocol *up=NULL; | |
ba9880c1 SS |
582 | |
583 | printf("Supported file protocols:\n"); | |
584 | while((up = av_protocol_next(up))) | |
2cb2d6f0 | 585 | printf("%s\n", up->name); |
ba9880c1 | 586 | } |
090b61b2 | 587 | |
62d75662 SS |
588 | void show_filters(void) |
589 | { | |
78638628 | 590 | AVFilter av_unused(**filter) = NULL; |
62d75662 SS |
591 | |
592 | printf("Filters:\n"); | |
663c2edf | 593 | #if CONFIG_AVFILTER |
62d75662 SS |
594 | while ((filter = av_filter_next(filter)) && *filter) |
595 | printf("%-16s %s\n", (*filter)->name, (*filter)->description); | |
663c2edf | 596 | #endif |
62d75662 SS |
597 | } |
598 | ||
3f7bb426 SS |
599 | void show_pix_fmts(void) |
600 | { | |
9cb5c760 SS |
601 | enum PixelFormat pix_fmt; |
602 | ||
603 | printf( | |
604 | "Pixel formats:\n" | |
605 | "I.... = Supported Input format for conversion\n" | |
606 | ".O... = Supported Output format for conversion\n" | |
607 | "..H.. = Hardware accelerated format\n" | |
608 | "...P. = Paletted format\n" | |
609 | "....B = Bitstream format\n" | |
610 | "FLAGS NAME NB_COMPONENTS BITS_PER_PIXEL\n" | |
611 | "-----\n"); | |
612 | ||
e24db3e3 MR |
613 | #if !CONFIG_SWSCALE |
614 | # define sws_isSupportedInput(x) 0 | |
615 | # define sws_isSupportedOutput(x) 0 | |
616 | #endif | |
617 | ||
9cb5c760 SS |
618 | for (pix_fmt = 0; pix_fmt < PIX_FMT_NB; pix_fmt++) { |
619 | const AVPixFmtDescriptor *pix_desc = &av_pix_fmt_descriptors[pix_fmt]; | |
620 | printf("%c%c%c%c%c %-16s %d %2d\n", | |
621 | sws_isSupportedInput (pix_fmt) ? 'I' : '.', | |
622 | sws_isSupportedOutput(pix_fmt) ? 'O' : '.', | |
623 | pix_desc->flags & PIX_FMT_HWACCEL ? 'H' : '.', | |
624 | pix_desc->flags & PIX_FMT_PAL ? 'P' : '.', | |
625 | pix_desc->flags & PIX_FMT_BITSTREAM ? 'B' : '.', | |
626 | pix_desc->name, | |
627 | pix_desc->nb_components, | |
628 | av_get_bits_per_pixel(pix_desc)); | |
629 | } | |
3f7bb426 SS |
630 | } |
631 | ||
090b61b2 SS |
632 | int read_yesno(void) |
633 | { | |
634 | int c = getchar(); | |
635 | int yesno = (toupper(c) == 'Y'); | |
636 | ||
637 | while (c != '\n' && c != EOF) | |
638 | c = getchar(); | |
639 | ||
640 | return yesno; | |
641 | } | |
458b062d SS |
642 | |
643 | int read_file(const char *filename, char **bufptr, size_t *size) | |
644 | { | |
c56e9e05 | 645 | FILE *f = fopen(filename, "rb"); |
458b062d SS |
646 | |
647 | if (!f) { | |
648 | fprintf(stderr, "Cannot read file '%s': %s\n", filename, strerror(errno)); | |
649 | return AVERROR(errno); | |
650 | } | |
651 | fseek(f, 0, SEEK_END); | |
652 | *size = ftell(f); | |
653 | fseek(f, 0, SEEK_SET); | |
654 | *bufptr = av_malloc(*size + 1); | |
655 | if (!*bufptr) { | |
656 | fprintf(stderr, "Could not allocate file buffer\n"); | |
657 | fclose(f); | |
658 | return AVERROR(ENOMEM); | |
659 | } | |
660 | fread(*bufptr, 1, *size, f); | |
661 | (*bufptr)[*size++] = '\0'; | |
662 | ||
663 | fclose(f); | |
664 | return 0; | |
665 | } |