vpxenc.c 83.1 KB
Newer Older
John Koleszar's avatar
John Koleszar committed
1
/*
2
 *  Copyright (c) 2010 The WebM project authors. All Rights Reserved.
John Koleszar's avatar
John Koleszar committed
3
 *
4
 *  Use of this source code is governed by a BSD-style license
5 6
 *  that can be found in the LICENSE file in the root of the source
 *  tree. An additional intellectual property rights grant can be found
7
 *  in the file PATENTS.  All contributing project authors may
8
 *  be found in the AUTHORS file in the root of the source tree.
John Koleszar's avatar
John Koleszar committed
9 10
 */

11
#include "./vpxenc.h"
12
#include "./vpx_config.h"
John Koleszar's avatar
John Koleszar committed
13

14 15
#include <assert.h>
#include <limits.h>
16
#include <math.h>
17
#include <stdarg.h>
John Koleszar's avatar
John Koleszar committed
18 19 20
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
21

22 23 24 25
#if CONFIG_LIBYUV
#include "third_party/libyuv/include/libyuv/scale.h"
#endif

26
#include "vpx/vpx_encoder.h"
27
#if CONFIG_DECODERS
John Koleszar's avatar
John Koleszar committed
28
#include "vpx/vpx_decoder.h"
29
#endif
John Koleszar's avatar
John Koleszar committed
30

31 32
#include "./args.h"
#include "./ivfenc.h"
Tom Finegan's avatar
Tom Finegan committed
33
#include "./tools_common.h"
34

35
#if CONFIG_VP8_ENCODER || CONFIG_VP9_ENCODER
36
#include "vpx/vp8cx.h"
John Koleszar's avatar
John Koleszar committed
37
#endif
38
#if CONFIG_VP8_DECODER || CONFIG_VP9_DECODER
39
#include "vpx/vp8dx.h"
John Koleszar's avatar
John Koleszar committed
40 41
#endif

Tom Finegan's avatar
Tom Finegan committed
42
#include "vpx/vpx_integer.h"
John Koleszar's avatar
John Koleszar committed
43 44
#include "vpx_ports/mem_ops.h"
#include "vpx_ports/vpx_timer.h"
45
#include "./rate_hist.h"
46
#include "./vpxstats.h"
47
#include "./warnings.h"
48
#if CONFIG_WEBM_IO
49
#include "./webmenc.h"
50
#endif
51
#include "./y4minput.h"
52

John Koleszar's avatar
John Koleszar committed
53 54
/* Swallow warnings about unused results of fread/fwrite */
static size_t wrap_fread(void *ptr, size_t size, size_t nmemb,
John Koleszar's avatar
John Koleszar committed
55 56
                         FILE *stream) {
  return fread(ptr, size, nmemb, stream);
John Koleszar's avatar
John Koleszar committed
57 58 59 60
}
#define fread wrap_fread

static size_t wrap_fwrite(const void *ptr, size_t size, size_t nmemb,
John Koleszar's avatar
John Koleszar committed
61 62
                          FILE *stream) {
  return fwrite(ptr, size, nmemb, stream);
John Koleszar's avatar
John Koleszar committed
63 64 65 66
}
#define fwrite wrap_fwrite


John Koleszar's avatar
John Koleszar committed
67 68
static const char *exec_name;

69 70
static void warn_or_exit_on_errorv(vpx_codec_ctx_t *ctx, int fatal,
                                   const char *s, va_list ap) {
John Koleszar's avatar
John Koleszar committed
71 72
  if (ctx->err) {
    const char *detail = vpx_codec_error_detail(ctx);
John Koleszar's avatar
John Koleszar committed
73

John Koleszar's avatar
John Koleszar committed
74 75
    vfprintf(stderr, s, ap);
    fprintf(stderr, ": %s\n", vpx_codec_error(ctx));
John Koleszar's avatar
John Koleszar committed
76

John Koleszar's avatar
John Koleszar committed
77 78
    if (detail)
      fprintf(stderr, "    %s\n", detail);
John Koleszar's avatar
John Koleszar committed
79

80 81
    if (fatal)
      exit(EXIT_FAILURE);
John Koleszar's avatar
John Koleszar committed
82
  }
John Koleszar's avatar
John Koleszar committed
83 84
}

85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
static void ctx_exit_on_error(vpx_codec_ctx_t *ctx, const char *s, ...) {
  va_list ap;

  va_start(ap, s);
  warn_or_exit_on_errorv(ctx, 1, s, ap);
  va_end(ap);
}

static void warn_or_exit_on_error(vpx_codec_ctx_t *ctx, int fatal,
                                  const char *s, ...) {
  va_list ap;

  va_start(ap, s);
  warn_or_exit_on_errorv(ctx, fatal, s, ap);
  va_end(ap);
}

102 103 104
int read_frame(struct VpxInputContext *input_ctx, vpx_image_t *img) {
  FILE *f = input_ctx->file;
  y4m_input *y4m = &input_ctx->y4m;
John Koleszar's avatar
John Koleszar committed
105 106
  int shortread = 0;

107
  if (input_ctx->file_type == FILE_TYPE_Y4M) {
John Koleszar's avatar
John Koleszar committed
108 109 110
    if (y4m_input_fetch_frame(y4m, f, img) < 1)
      return 0;
  } else {
111
    shortread = read_yuv_frame(input_ctx, img);
John Koleszar's avatar
John Koleszar committed
112
  }
John Koleszar's avatar
John Koleszar committed
113

John Koleszar's avatar
John Koleszar committed
114
  return !shortread;
John Koleszar's avatar
John Koleszar committed
115 116
}

117
int file_is_y4m(const char detect[4]) {
John Koleszar's avatar
John Koleszar committed
118 119 120 121
  if (memcmp(detect, "YUV4", 4) == 0) {
    return 1;
  }
  return 0;
122 123
}

124 125 126 127 128 129
int fourcc_is_ivf(const char detect[4]) {
  if (memcmp(detect, "DKIF", 4) == 0) {
    return 1;
  }
  return 0;
}
John Koleszar's avatar
John Koleszar committed
130

131
static const arg_def_t debugmode = ARG_DEF("D", "debug", 0,
John Koleszar's avatar
John Koleszar committed
132
                                           "Debug mode (makes output deterministic)");
133
static const arg_def_t outputfile = ARG_DEF("o", "output", 1,
John Koleszar's avatar
John Koleszar committed
134
                                            "Output filename");
John Koleszar's avatar
John Koleszar committed
135
static const arg_def_t use_yv12 = ARG_DEF(NULL, "yv12", 0,
John Koleszar's avatar
John Koleszar committed
136
                                          "Input file is YV12 ");
John Koleszar's avatar
John Koleszar committed
137
static const arg_def_t use_i420 = ARG_DEF(NULL, "i420", 0,
John Koleszar's avatar
John Koleszar committed
138
                                          "Input file is I420 (default)");
139 140 141 142
static const arg_def_t use_i422 = ARG_DEF(NULL, "i422", 0,
                                          "Input file is I422");
static const arg_def_t use_i444 = ARG_DEF(NULL, "i444", 0,
                                          "Input file is I444");
Deb Mukherjee's avatar
Deb Mukherjee committed
143 144
static const arg_def_t use_i440 = ARG_DEF(NULL, "i440", 0,
                                          "Input file is I440");
John Koleszar's avatar
John Koleszar committed
145
static const arg_def_t codecarg = ARG_DEF(NULL, "codec", 1,
John Koleszar's avatar
John Koleszar committed
146
                                          "Codec to use");
John Koleszar's avatar
John Koleszar committed
147
static const arg_def_t passes           = ARG_DEF("p", "passes", 1,
John Koleszar's avatar
John Koleszar committed
148
                                                  "Number of passes (1/2)");
John Koleszar's avatar
John Koleszar committed
149
static const arg_def_t pass_arg         = ARG_DEF(NULL, "pass", 1,
John Koleszar's avatar
John Koleszar committed
150
                                                  "Pass to execute (1/2)");
John Koleszar's avatar
John Koleszar committed
151
static const arg_def_t fpf_name         = ARG_DEF(NULL, "fpf", 1,
John Koleszar's avatar
John Koleszar committed
152
                                                  "First pass statistics file name");
153 154 155 156
#if CONFIG_FP_MB_STATS
static const arg_def_t fpmbf_name         = ARG_DEF(NULL, "fpmbf", 1,
                                      "First pass block statistics file name");
#endif
John Koleszar's avatar
John Koleszar committed
157 158
static const arg_def_t limit = ARG_DEF(NULL, "limit", 1,
                                       "Stop encoding after n input frames");
159
static const arg_def_t skip = ARG_DEF(NULL, "skip", 1,
John Koleszar's avatar
John Koleszar committed
160
                                      "Skip the first n input frames");
John Koleszar's avatar
John Koleszar committed
161
static const arg_def_t deadline         = ARG_DEF("d", "deadline", 1,
John Koleszar's avatar
John Koleszar committed
162
                                                  "Deadline per frame (usec)");
John Koleszar's avatar
John Koleszar committed
163
static const arg_def_t best_dl          = ARG_DEF(NULL, "best", 0,
John Koleszar's avatar
John Koleszar committed
164
                                                  "Use Best Quality Deadline");
John Koleszar's avatar
John Koleszar committed
165
static const arg_def_t good_dl          = ARG_DEF(NULL, "good", 0,
John Koleszar's avatar
John Koleszar committed
166
                                                  "Use Good Quality Deadline");
John Koleszar's avatar
John Koleszar committed
167
static const arg_def_t rt_dl            = ARG_DEF(NULL, "rt", 0,
John Koleszar's avatar
John Koleszar committed
168
                                                  "Use Realtime Quality Deadline");
James Zern's avatar
James Zern committed
169
static const arg_def_t quietarg         = ARG_DEF("q", "quiet", 0,
John Koleszar's avatar
John Koleszar committed
170
                                                  "Do not print encode progress");
John Koleszar's avatar
John Koleszar committed
171
static const arg_def_t verbosearg       = ARG_DEF("v", "verbose", 0,
John Koleszar's avatar
John Koleszar committed
172
                                                  "Show encoder parameters");
John Koleszar's avatar
John Koleszar committed
173
static const arg_def_t psnrarg          = ARG_DEF(NULL, "psnr", 0,
John Koleszar's avatar
John Koleszar committed
174
                                                  "Show PSNR in status line");
175

176 177 178 179 180 181 182 183 184
static const struct arg_enum_list test_decode_enum[] = {
  {"off",   TEST_DECODE_OFF},
  {"fatal", TEST_DECODE_FATAL},
  {"warn",  TEST_DECODE_WARN},
  {NULL, 0}
};
static const arg_def_t recontest = ARG_DEF_ENUM(NULL, "test-decode", 1,
                                                "Test encode/decode mismatch",
                                                test_decode_enum);
185
static const arg_def_t framerate        = ARG_DEF(NULL, "fps", 1,
John Koleszar's avatar
John Koleszar committed
186
                                                  "Stream frame rate (rate/scale)");
John Koleszar's avatar
John Koleszar committed
187
static const arg_def_t use_ivf          = ARG_DEF(NULL, "ivf", 0,
188
                                                  "Output IVF (default is WebM if WebM IO is enabled)");
189
static const arg_def_t out_part = ARG_DEF("P", "output-partitions", 0,
John Koleszar's avatar
John Koleszar committed
190
                                          "Makes encoder output partitions. Requires IVF output!");
191
static const arg_def_t q_hist_n         = ARG_DEF(NULL, "q-hist", 1,
John Koleszar's avatar
John Koleszar committed
192
                                                  "Show quantizer histogram (n-buckets)");
193
static const arg_def_t rate_hist_n         = ARG_DEF(NULL, "rate-hist", 1,
John Koleszar's avatar
John Koleszar committed
194
                                                     "Show rate histogram (n-buckets)");
195 196 197
static const arg_def_t disable_warnings =
    ARG_DEF(NULL, "disable-warnings", 0,
            "Disable warnings about potentially incorrect encode settings.");
198 199 200
static const arg_def_t disable_warning_prompt =
    ARG_DEF("y", "disable-warning-prompt", 0,
            "Display warnings, but do not prompt user to continue.");
201 202 203 204
static const arg_def_t experimental_bitstream =
    ARG_DEF(NULL, "experimental-bitstream", 0,
            "Allow experimental bitstream features.");

205 206 207 208
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
static const arg_def_t test16bitinternalarg = ARG_DEF(
    NULL, "test-16bit-internal", 0, "Force use of 16 bit internal buffer");
#endif
209

John Koleszar's avatar
John Koleszar committed
210 211 212
static const arg_def_t *main_args[] = {
  &debugmode,
  &outputfile, &codecarg, &passes, &pass_arg, &fpf_name, &limit, &skip,
John Koleszar's avatar
John Koleszar committed
213
  &deadline, &best_dl, &good_dl, &rt_dl,
214
  &quietarg, &verbosearg, &psnrarg, &use_ivf, &out_part, &q_hist_n,
215
  &rate_hist_n, &disable_warnings, &disable_warning_prompt,
John Koleszar's avatar
John Koleszar committed
216
  NULL
John Koleszar's avatar
John Koleszar committed
217 218 219
};

static const arg_def_t usage            = ARG_DEF("u", "usage", 1,
John Koleszar's avatar
John Koleszar committed
220
                                                  "Usage profile number to use");
John Koleszar's avatar
John Koleszar committed
221
static const arg_def_t threads          = ARG_DEF("t", "threads", 1,
John Koleszar's avatar
John Koleszar committed
222
                                                  "Max number of threads to use");
John Koleszar's avatar
John Koleszar committed
223
static const arg_def_t profile          = ARG_DEF(NULL, "profile", 1,
John Koleszar's avatar
John Koleszar committed
224
                                                  "Bitstream profile number to use");
John Koleszar's avatar
John Koleszar committed
225
static const arg_def_t width            = ARG_DEF("w", "width", 1,
John Koleszar's avatar
John Koleszar committed
226
                                                  "Frame width");
John Koleszar's avatar
John Koleszar committed
227
static const arg_def_t height           = ARG_DEF("h", "height", 1,
John Koleszar's avatar
John Koleszar committed
228
                                                  "Frame height");
229
#if CONFIG_WEBM_IO
230
static const struct arg_enum_list stereo_mode_enum[] = {
John Koleszar's avatar
John Koleszar committed
231 232 233 234 235 236
  {"mono", STEREO_FORMAT_MONO},
  {"left-right", STEREO_FORMAT_LEFT_RIGHT},
  {"bottom-top", STEREO_FORMAT_BOTTOM_TOP},
  {"top-bottom", STEREO_FORMAT_TOP_BOTTOM},
  {"right-left", STEREO_FORMAT_RIGHT_LEFT},
  {NULL, 0}
237 238
};
static const arg_def_t stereo_mode      = ARG_DEF_ENUM(NULL, "stereo-mode", 1,
John Koleszar's avatar
John Koleszar committed
239
                                                       "Stereo 3D video format", stereo_mode_enum);
240
#endif
John Koleszar's avatar
John Koleszar committed
241
static const arg_def_t timebase         = ARG_DEF(NULL, "timebase", 1,
John Koleszar's avatar
John Koleszar committed
242
                                                  "Output timestamp precision (fractional seconds)");
John Koleszar's avatar
John Koleszar committed
243
static const arg_def_t error_resilient  = ARG_DEF(NULL, "error-resilient", 1,
John Koleszar's avatar
John Koleszar committed
244
                                                  "Enable error resiliency features");
John Koleszar's avatar
John Koleszar committed
245
static const arg_def_t lag_in_frames    = ARG_DEF(NULL, "lag-in-frames", 1,
John Koleszar's avatar
John Koleszar committed
246
                                                  "Max number of frames to lag");
John Koleszar's avatar
John Koleszar committed
247

John Koleszar's avatar
John Koleszar committed
248
static const arg_def_t *global_args[] = {
Deb Mukherjee's avatar
Deb Mukherjee committed
249
  &use_yv12, &use_i420, &use_i422, &use_i444, &use_i440,
250
  &usage, &threads, &profile,
251 252 253 254 255
  &width, &height,
#if CONFIG_WEBM_IO
  &stereo_mode,
#endif
  &timebase, &framerate,
256
  &error_resilient,
257 258 259
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
  &test16bitinternalarg,
#endif
John Koleszar's avatar
John Koleszar committed
260
  &lag_in_frames, NULL
John Koleszar's avatar
John Koleszar committed
261 262 263
};

static const arg_def_t dropframe_thresh   = ARG_DEF(NULL, "drop-frame", 1,
John Koleszar's avatar
John Koleszar committed
264
                                                    "Temporal resampling threshold (buf %)");
John Koleszar's avatar
John Koleszar committed
265
static const arg_def_t resize_allowed     = ARG_DEF(NULL, "resize-allowed", 1,
John Koleszar's avatar
John Koleszar committed
266
                                                    "Spatial resampling enabled (bool)");
267 268 269 270
static const arg_def_t resize_width       = ARG_DEF(NULL, "resize-width", 1,
                                                    "Width of encoded frame");
static const arg_def_t resize_height      = ARG_DEF(NULL, "resize-height", 1,
                                                    "Height of encoded frame");
John Koleszar's avatar
John Koleszar committed
271
static const arg_def_t resize_up_thresh   = ARG_DEF(NULL, "resize-up", 1,
John Koleszar's avatar
John Koleszar committed
272
                                                    "Upscale threshold (buf %)");
John Koleszar's avatar
John Koleszar committed
273
static const arg_def_t resize_down_thresh = ARG_DEF(NULL, "resize-down", 1,
John Koleszar's avatar
John Koleszar committed
274
                                                    "Downscale threshold (buf %)");
275
static const struct arg_enum_list end_usage_enum[] = {
John Koleszar's avatar
John Koleszar committed
276 277 278
  {"vbr", VPX_VBR},
  {"cbr", VPX_CBR},
  {"cq",  VPX_CQ},
279
  {"q",   VPX_Q},
John Koleszar's avatar
John Koleszar committed
280
  {NULL, 0}
281 282
};
static const arg_def_t end_usage          = ARG_DEF_ENUM(NULL, "end-usage", 1,
John Koleszar's avatar
John Koleszar committed
283
                                                         "Rate control mode", end_usage_enum);
John Koleszar's avatar
John Koleszar committed
284
static const arg_def_t target_bitrate     = ARG_DEF(NULL, "target-bitrate", 1,
John Koleszar's avatar
John Koleszar committed
285
                                                    "Bitrate (kbps)");
John Koleszar's avatar
John Koleszar committed
286
static const arg_def_t min_quantizer      = ARG_DEF(NULL, "min-q", 1,
John Koleszar's avatar
John Koleszar committed
287
                                                    "Minimum (best) quantizer");
John Koleszar's avatar
John Koleszar committed
288
static const arg_def_t max_quantizer      = ARG_DEF(NULL, "max-q", 1,
John Koleszar's avatar
John Koleszar committed
289
                                                    "Maximum (worst) quantizer");
John Koleszar's avatar
John Koleszar committed
290
static const arg_def_t undershoot_pct     = ARG_DEF(NULL, "undershoot-pct", 1,
John Koleszar's avatar
John Koleszar committed
291
                                                    "Datarate undershoot (min) target (%)");
John Koleszar's avatar
John Koleszar committed
292
static const arg_def_t overshoot_pct      = ARG_DEF(NULL, "overshoot-pct", 1,
John Koleszar's avatar
John Koleszar committed
293
                                                    "Datarate overshoot (max) target (%)");
John Koleszar's avatar
John Koleszar committed
294
static const arg_def_t buf_sz             = ARG_DEF(NULL, "buf-sz", 1,
John Koleszar's avatar
John Koleszar committed
295
                                                    "Client buffer size (ms)");
John Koleszar's avatar
John Koleszar committed
296
static const arg_def_t buf_initial_sz     = ARG_DEF(NULL, "buf-initial-sz", 1,
John Koleszar's avatar
John Koleszar committed
297
                                                    "Client initial buffer size (ms)");
John Koleszar's avatar
John Koleszar committed
298
static const arg_def_t buf_optimal_sz     = ARG_DEF(NULL, "buf-optimal-sz", 1,
John Koleszar's avatar
John Koleszar committed
299 300
                                                    "Client optimal buffer size (ms)");
static const arg_def_t *rc_args[] = {
301 302 303 304
  &dropframe_thresh, &resize_allowed, &resize_width, &resize_height,
  &resize_up_thresh, &resize_down_thresh, &end_usage, &target_bitrate,
  &min_quantizer, &max_quantizer, &undershoot_pct, &overshoot_pct, &buf_sz,
  &buf_initial_sz, &buf_optimal_sz, NULL
John Koleszar's avatar
John Koleszar committed
305 306 307 308
};


static const arg_def_t bias_pct = ARG_DEF(NULL, "bias-pct", 1,
John Koleszar's avatar
John Koleszar committed
309
                                          "CBR/VBR bias (0=CBR, 100=VBR)");
John Koleszar's avatar
John Koleszar committed
310
static const arg_def_t minsection_pct = ARG_DEF(NULL, "minsection-pct", 1,
John Koleszar's avatar
John Koleszar committed
311
                                                "GOP min bitrate (% of target)");
John Koleszar's avatar
John Koleszar committed
312
static const arg_def_t maxsection_pct = ARG_DEF(NULL, "maxsection-pct", 1,
John Koleszar's avatar
John Koleszar committed
313 314 315
                                                "GOP max bitrate (% of target)");
static const arg_def_t *rc_twopass_args[] = {
  &bias_pct, &minsection_pct, &maxsection_pct, NULL
John Koleszar's avatar
John Koleszar committed
316 317 318 319
};


static const arg_def_t kf_min_dist = ARG_DEF(NULL, "kf-min-dist", 1,
John Koleszar's avatar
John Koleszar committed
320
                                             "Minimum keyframe interval (frames)");
John Koleszar's avatar
John Koleszar committed
321
static const arg_def_t kf_max_dist = ARG_DEF(NULL, "kf-max-dist", 1,
John Koleszar's avatar
John Koleszar committed
322
                                             "Maximum keyframe interval (frames)");
323
static const arg_def_t kf_disabled = ARG_DEF(NULL, "disable-kf", 0,
John Koleszar's avatar
John Koleszar committed
324 325 326
                                             "Disable keyframe placement");
static const arg_def_t *kf_args[] = {
  &kf_min_dist, &kf_max_dist, &kf_disabled, NULL
John Koleszar's avatar
John Koleszar committed
327 328 329 330
};


static const arg_def_t noise_sens = ARG_DEF(NULL, "noise-sensitivity", 1,
John Koleszar's avatar
John Koleszar committed
331
                                            "Noise sensitivity (frames to blur)");
John Koleszar's avatar
John Koleszar committed
332
static const arg_def_t sharpness = ARG_DEF(NULL, "sharpness", 1,
333
                                           "Loop filter sharpness (0..7)");
John Koleszar's avatar
John Koleszar committed
334
static const arg_def_t static_thresh = ARG_DEF(NULL, "static-thresh", 1,
John Koleszar's avatar
John Koleszar committed
335
                                               "Motion detection threshold");
John Koleszar's avatar
John Koleszar committed
336
static const arg_def_t cpu_used = ARG_DEF(NULL, "cpu-used", 1,
John Koleszar's avatar
John Koleszar committed
337
                                          "CPU Used (-16..16)");
John Koleszar's avatar
John Koleszar committed
338
static const arg_def_t auto_altref = ARG_DEF(NULL, "auto-alt-ref", 1,
John Koleszar's avatar
John Koleszar committed
339
                                             "Enable automatic alt reference frames");
John Koleszar's avatar
John Koleszar committed
340
static const arg_def_t arnr_maxframes = ARG_DEF(NULL, "arnr-maxframes", 1,
341
                                                "AltRef max frames (0..15)");
John Koleszar's avatar
John Koleszar committed
342
static const arg_def_t arnr_strength = ARG_DEF(NULL, "arnr-strength", 1,
343
                                               "AltRef filter strength (0..6)");
John Koleszar's avatar
John Koleszar committed
344
static const arg_def_t arnr_type = ARG_DEF(NULL, "arnr-type", 1,
345
                                           "AltRef type");
John Koleszar's avatar
John Koleszar committed
346
static const struct arg_enum_list tuning_enum[] = {
John Koleszar's avatar
John Koleszar committed
347 348 349
  {"psnr", VP8_TUNE_PSNR},
  {"ssim", VP8_TUNE_SSIM},
  {NULL, 0}
John Koleszar's avatar
John Koleszar committed
350 351
};
static const arg_def_t tune_ssim = ARG_DEF_ENUM(NULL, "tune", 1,
John Koleszar's avatar
John Koleszar committed
352
                                                "Material to favor", tuning_enum);
Paul Wilkins's avatar
CQ Mode  
Paul Wilkins committed
353
static const arg_def_t cq_level = ARG_DEF(NULL, "cq-level", 1,
354
                                          "Constant/Constrained Quality level");
355
static const arg_def_t max_intra_rate_pct = ARG_DEF(NULL, "max-intra-rate", 1,
John Koleszar's avatar
John Koleszar committed
356
                                                    "Max I-frame bitrate (pct)");
John Koleszar's avatar
John Koleszar committed
357

358
#if CONFIG_VP8_ENCODER
359 360
static const arg_def_t token_parts =
    ARG_DEF(NULL, "token-parts", 1, "Number of token partitions to use, log2");
John Koleszar's avatar
John Koleszar committed
361 362 363
static const arg_def_t *vp8_args[] = {
  &cpu_used, &auto_altref, &noise_sens, &sharpness, &static_thresh,
  &token_parts, &arnr_maxframes, &arnr_strength, &arnr_type,
John Koleszar's avatar
John Koleszar committed
364
  &tune_ssim, &cq_level, &max_intra_rate_pct,
365
  NULL
John Koleszar's avatar
John Koleszar committed
366
};
367 368 369 370 371 372 373 374 375 376 377
static const int vp8_arg_ctrl_map[] = {
  VP8E_SET_CPUUSED, VP8E_SET_ENABLEAUTOALTREF,
  VP8E_SET_NOISE_SENSITIVITY, VP8E_SET_SHARPNESS, VP8E_SET_STATIC_THRESHOLD,
  VP8E_SET_TOKEN_PARTITIONS,
  VP8E_SET_ARNR_MAXFRAMES, VP8E_SET_ARNR_STRENGTH, VP8E_SET_ARNR_TYPE,
  VP8E_SET_TUNING, VP8E_SET_CQ_LEVEL, VP8E_SET_MAX_INTRA_BITRATE_PCT,
  0
};
#endif

#if CONFIG_VP9_ENCODER
378 379 380 381 382 383 384 385 386
static const arg_def_t tile_cols =
    ARG_DEF(NULL, "tile-columns", 1, "Number of tile columns to use, log2");
static const arg_def_t tile_rows =
    ARG_DEF(NULL, "tile-rows", 1, "Number of tile rows to use, log2");
static const arg_def_t lossless = ARG_DEF(NULL, "lossless", 1, "Lossless mode");
static const arg_def_t frame_parallel_decoding = ARG_DEF(
    NULL, "frame-parallel", 1, "Enable frame parallel decodability features");
static const arg_def_t aq_mode = ARG_DEF(
    NULL, "aq-mode", 1,
387
    "Adaptive quantization mode (0: off (default), 1: variance 2: complexity, "
388 389
    "3: cyclic refresh)");
static const arg_def_t frame_periodic_boost = ARG_DEF(
390
    NULL, "frame-boost", 1,
391
    "Enable frame periodic boost (0: off (default), 1: on)");
392

393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
static const struct arg_enum_list bitdepth_enum[] = {
  {"8",  VPX_BITS_8},
  {"10", VPX_BITS_10},
  {"12", VPX_BITS_12},
  {NULL, 0}
};

static const arg_def_t bitdeptharg   = ARG_DEF_ENUM("b", "bit-depth", 1,
                                                    "Bit depth for codec "
                                                    "(8 for version <=1, "
                                                    "10 or 12 for version 2)",
                                                    bitdepth_enum);
static const arg_def_t inbitdeptharg = ARG_DEF(NULL, "input-bit-depth", 1,
                                               "Bit depth of input");
#endif

410 411 412 413 414 415 416 417 418
static const struct arg_enum_list tune_content_enum[] = {
  {"default", VP9E_CONTENT_DEFAULT},
  {"screen", VP9E_CONTENT_SCREEN},
  {NULL, 0}
};

static const arg_def_t tune_content = ARG_DEF_ENUM(
    NULL, "tune-content", 1, "Tune content type", tune_content_enum);

419
static const arg_def_t *vp9_args[] = {
420
  &cpu_used, &auto_altref, &sharpness, &static_thresh,
421
  &tile_cols, &tile_rows, &arnr_maxframes, &arnr_strength, &arnr_type,
Yaowu Xu's avatar
Yaowu Xu committed
422
  &tune_ssim, &cq_level, &max_intra_rate_pct, &lossless,
423 424
  &frame_parallel_decoding, &aq_mode, &frame_periodic_boost,
  &noise_sens, &tune_content,
425 426 427
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
  &bitdeptharg, &inbitdeptharg,
#endif
John Koleszar's avatar
John Koleszar committed
428
  NULL
John Koleszar's avatar
John Koleszar committed
429
};
430
static const int vp9_arg_ctrl_map[] = {
John Koleszar's avatar
John Koleszar committed
431
  VP8E_SET_CPUUSED, VP8E_SET_ENABLEAUTOALTREF,
432
  VP8E_SET_SHARPNESS, VP8E_SET_STATIC_THRESHOLD,
Ronald S. Bultje's avatar
Ronald S. Bultje committed
433
  VP9E_SET_TILE_COLUMNS, VP9E_SET_TILE_ROWS,
434
  VP8E_SET_ARNR_MAXFRAMES, VP8E_SET_ARNR_STRENGTH, VP8E_SET_ARNR_TYPE,
John Koleszar's avatar
John Koleszar committed
435
  VP8E_SET_TUNING, VP8E_SET_CQ_LEVEL, VP8E_SET_MAX_INTRA_BITRATE_PCT,
436
  VP9E_SET_LOSSLESS, VP9E_SET_FRAME_PARALLEL_DECODING, VP9E_SET_AQ_MODE,
437 438
  VP9E_SET_FRAME_PERIODIC_BOOST, VP9E_SET_NOISE_SENSITIVITY,
  VP9E_SET_TUNE_CONTENT,
John Koleszar's avatar
John Koleszar committed
439
  0
John Koleszar's avatar
John Koleszar committed
440 441 442 443 444
};
#endif

static const arg_def_t *no_args[] = { NULL };

445
void usage_exit() {
John Koleszar's avatar
John Koleszar committed
446 447 448 449 450 451
  int i;

  fprintf(stderr, "Usage: %s <options> -o dst_filename src_filename \n",
          exec_name);

  fprintf(stderr, "\nOptions:\n");
John Koleszar's avatar
John Koleszar committed
452
  arg_show_usage(stderr, main_args);
John Koleszar's avatar
John Koleszar committed
453
  fprintf(stderr, "\nEncoder Global Options:\n");
John Koleszar's avatar
John Koleszar committed
454
  arg_show_usage(stderr, global_args);
John Koleszar's avatar
John Koleszar committed
455
  fprintf(stderr, "\nRate Control Options:\n");
John Koleszar's avatar
John Koleszar committed
456
  arg_show_usage(stderr, rc_args);
John Koleszar's avatar
John Koleszar committed
457
  fprintf(stderr, "\nTwopass Rate Control Options:\n");
John Koleszar's avatar
John Koleszar committed
458
  arg_show_usage(stderr, rc_twopass_args);
John Koleszar's avatar
John Koleszar committed
459
  fprintf(stderr, "\nKeyframe Placement Options:\n");
John Koleszar's avatar
John Koleszar committed
460
  arg_show_usage(stderr, kf_args);
John Koleszar's avatar
John Koleszar committed
461
#if CONFIG_VP8_ENCODER
John Koleszar's avatar
John Koleszar committed
462
  fprintf(stderr, "\nVP8 Specific Options:\n");
John Koleszar's avatar
John Koleszar committed
463
  arg_show_usage(stderr, vp8_args);
John Koleszar's avatar
John Koleszar committed
464
#endif
465 466
#if CONFIG_VP9_ENCODER
  fprintf(stderr, "\nVP9 Specific Options:\n");
John Koleszar's avatar
John Koleszar committed
467
  arg_show_usage(stderr, vp9_args);
John Koleszar's avatar
John Koleszar committed
468
#endif
John Koleszar's avatar
John Koleszar committed
469 470 471
  fprintf(stderr, "\nStream timebase (--timebase):\n"
          "  The desired precision of timestamps in the output, expressed\n"
          "  in fractional seconds. Default is 1/1000.\n");
472
  fprintf(stderr, "\nIncluded encoders:\n\n");
John Koleszar's avatar
John Koleszar committed
473

474 475
  for (i = 0; i < get_vpx_encoder_count(); ++i) {
    const VpxInterface *const encoder = get_vpx_encoder_by_index(i);
John Koleszar's avatar
John Koleszar committed
476
    fprintf(stderr, "    %-6s - %s\n",
477
            encoder->name, vpx_codec_iface_name(encoder->codec_interface()));
478
  }
John Koleszar's avatar
John Koleszar committed
479 480

  exit(EXIT_FAILURE);
John Koleszar's avatar
John Koleszar committed
481 482
}

483
#define mmin(a, b)  ((a) < (b) ? (a) : (b))
484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579

#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
static void find_mismatch_high(const vpx_image_t *const img1,
                               const vpx_image_t *const img2,
                               int yloc[4], int uloc[4], int vloc[4]) {
  uint16_t *plane1, *plane2;
  uint32_t stride1, stride2;
  const uint32_t bsize = 64;
  const uint32_t bsizey = bsize >> img1->y_chroma_shift;
  const uint32_t bsizex = bsize >> img1->x_chroma_shift;
  const uint32_t c_w =
      (img1->d_w + img1->x_chroma_shift) >> img1->x_chroma_shift;
  const uint32_t c_h =
      (img1->d_h + img1->y_chroma_shift) >> img1->y_chroma_shift;
  int match = 1;
  uint32_t i, j;
  yloc[0] = yloc[1] = yloc[2] = yloc[3] = -1;
  plane1 = (uint16_t*)img1->planes[VPX_PLANE_Y];
  plane2 = (uint16_t*)img2->planes[VPX_PLANE_Y];
  stride1 = img1->stride[VPX_PLANE_Y]/2;
  stride2 = img2->stride[VPX_PLANE_Y]/2;
  for (i = 0, match = 1; match && i < img1->d_h; i += bsize) {
    for (j = 0; match && j < img1->d_w; j += bsize) {
      int k, l;
      const int si = mmin(i + bsize, img1->d_h) - i;
      const int sj = mmin(j + bsize, img1->d_w) - j;
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
          if (*(plane1 + (i + k) * stride1 + j + l) !=
              *(plane2 + (i + k) * stride2 + j + l)) {
            yloc[0] = i + k;
            yloc[1] = j + l;
            yloc[2] = *(plane1 + (i + k) * stride1 + j + l);
            yloc[3] = *(plane2 + (i + k) * stride2 + j + l);
            match = 0;
            break;
          }
        }
      }
    }
  }

  uloc[0] = uloc[1] = uloc[2] = uloc[3] = -1;
  plane1 = (uint16_t*)img1->planes[VPX_PLANE_U];
  plane2 = (uint16_t*)img2->planes[VPX_PLANE_U];
  stride1 = img1->stride[VPX_PLANE_U]/2;
  stride2 = img2->stride[VPX_PLANE_U]/2;
  for (i = 0, match = 1; match && i < c_h; i += bsizey) {
    for (j = 0; match && j < c_w; j += bsizex) {
      int k, l;
      const int si = mmin(i + bsizey, c_h - i);
      const int sj = mmin(j + bsizex, c_w - j);
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
          if (*(plane1 + (i + k) * stride1 + j + l) !=
              *(plane2 + (i + k) * stride2 + j + l)) {
            uloc[0] = i + k;
            uloc[1] = j + l;
            uloc[2] = *(plane1 + (i + k) * stride1 + j + l);
            uloc[3] = *(plane2 + (i + k) * stride2 + j + l);
            match = 0;
            break;
          }
        }
      }
    }
  }

  vloc[0] = vloc[1] = vloc[2] = vloc[3] = -1;
  plane1 = (uint16_t*)img1->planes[VPX_PLANE_V];
  plane2 = (uint16_t*)img2->planes[VPX_PLANE_V];
  stride1 = img1->stride[VPX_PLANE_V]/2;
  stride2 = img2->stride[VPX_PLANE_V]/2;
  for (i = 0, match = 1; match && i < c_h; i += bsizey) {
    for (j = 0; match && j < c_w; j += bsizex) {
      int k, l;
      const int si = mmin(i + bsizey, c_h - i);
      const int sj = mmin(j + bsizex, c_w - j);
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
          if (*(plane1 + (i + k) * stride1 + j + l) !=
              *(plane2 + (i + k) * stride2 + j + l)) {
            vloc[0] = i + k;
            vloc[1] = j + l;
            vloc[2] = *(plane1 + (i + k) * stride1 + j + l);
            vloc[3] = *(plane2 + (i + k) * stride2 + j + l);
            match = 0;
            break;
          }
        }
      }
    }
  }
}
#endif

James Zern's avatar
James Zern committed
580 581
static void find_mismatch(const vpx_image_t *const img1,
                          const vpx_image_t *const img2,
582
                          int yloc[4], int uloc[4], int vloc[4]) {
583 584 585 586 587 588 589 590 591
  const uint32_t bsize = 64;
  const uint32_t bsizey = bsize >> img1->y_chroma_shift;
  const uint32_t bsizex = bsize >> img1->x_chroma_shift;
  const uint32_t c_w =
      (img1->d_w + img1->x_chroma_shift) >> img1->x_chroma_shift;
  const uint32_t c_h =
      (img1->d_h + img1->y_chroma_shift) >> img1->y_chroma_shift;
  int match = 1;
  uint32_t i, j;
592
  yloc[0] = yloc[1] = yloc[2] = yloc[3] = -1;
593 594
  for (i = 0, match = 1; match && i < img1->d_h; i += bsize) {
    for (j = 0; match && j < img1->d_w; j += bsize) {
595
      int k, l;
James Zern's avatar
James Zern committed
596 597 598 599
      const int si = mmin(i + bsize, img1->d_h) - i;
      const int sj = mmin(j + bsize, img1->d_w) - j;
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
600 601 602 603 604 605
          if (*(img1->planes[VPX_PLANE_Y] +
                (i + k) * img1->stride[VPX_PLANE_Y] + j + l) !=
              *(img2->planes[VPX_PLANE_Y] +
                (i + k) * img2->stride[VPX_PLANE_Y] + j + l)) {
            yloc[0] = i + k;
            yloc[1] = j + l;
606 607 608 609
            yloc[2] = *(img1->planes[VPX_PLANE_Y] +
                        (i + k) * img1->stride[VPX_PLANE_Y] + j + l);
            yloc[3] = *(img2->planes[VPX_PLANE_Y] +
                        (i + k) * img2->stride[VPX_PLANE_Y] + j + l);
610
            match = 0;
611
            break;
612 613
          }
        }
James Zern's avatar
James Zern committed
614
      }
615
    }
616
  }
617

618
  uloc[0] = uloc[1] = uloc[2] = uloc[3] = -1;
619
  for (i = 0, match = 1; match && i < c_h; i += bsizey) {
John Koleszar's avatar
John Koleszar committed
620
    for (j = 0; match && j < c_w; j += bsizex) {
621
      int k, l;
James Zern's avatar
James Zern committed
622 623 624 625
      const int si = mmin(i + bsizey, c_h - i);
      const int sj = mmin(j + bsizex, c_w - j);
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
626 627 628 629 630 631
          if (*(img1->planes[VPX_PLANE_U] +
                (i + k) * img1->stride[VPX_PLANE_U] + j + l) !=
              *(img2->planes[VPX_PLANE_U] +
                (i + k) * img2->stride[VPX_PLANE_U] + j + l)) {
            uloc[0] = i + k;
            uloc[1] = j + l;
632 633 634
            uloc[2] = *(img1->planes[VPX_PLANE_U] +
                        (i + k) * img1->stride[VPX_PLANE_U] + j + l);
            uloc[3] = *(img2->planes[VPX_PLANE_U] +
James Zern's avatar
James Zern committed
635
                        (i + k) * img2->stride[VPX_PLANE_U] + j + l);
636 637 638 639
            match = 0;
            break;
          }
        }
James Zern's avatar
James Zern committed
640
      }
641 642
    }
  }
643
  vloc[0] = vloc[1] = vloc[2] = vloc[3] = -1;
644
  for (i = 0, match = 1; match && i < c_h; i += bsizey) {
John Koleszar's avatar
John Koleszar committed
645
    for (j = 0; match && j < c_w; j += bsizex) {
646
      int k, l;
James Zern's avatar
James Zern committed
647 648 649 650
      const int si = mmin(i + bsizey, c_h - i);
      const int sj = mmin(j + bsizex, c_w - j);
      for (k = 0; match && k < si; ++k) {
        for (l = 0; match && l < sj; ++l) {
651 652 653 654 655 656
          if (*(img1->planes[VPX_PLANE_V] +
                (i + k) * img1->stride[VPX_PLANE_V] + j + l) !=
              *(img2->planes[VPX_PLANE_V] +
                (i + k) * img2->stride[VPX_PLANE_V] + j + l)) {
            vloc[0] = i + k;
            vloc[1] = j + l;
657 658 659 660
            vloc[2] = *(img1->planes[VPX_PLANE_V] +
                        (i + k) * img1->stride[VPX_PLANE_V] + j + l);
            vloc[3] = *(img2->planes[VPX_PLANE_V] +
                        (i + k) * img2->stride[VPX_PLANE_V] + j + l);
661 662 663 664
            match = 0;
            break;
          }
        }
James Zern's avatar
James Zern committed
665
      }
666 667
    }
  }
668 669
}

James Zern's avatar
James Zern committed
670 671
static int compare_img(const vpx_image_t *const img1,
                       const vpx_image_t *const img2) {
672 673
  uint32_t l_w = img1->d_w;
  uint32_t c_w =
674 675 676 677
      (img1->d_w + img1->x_chroma_shift) >> img1->x_chroma_shift;
  const uint32_t c_h =
      (img1->d_h + img1->y_chroma_shift) >> img1->y_chroma_shift;
  uint32_t i;
John Koleszar's avatar
John Koleszar committed
678
  int match = 1;
679

John Koleszar's avatar
John Koleszar committed
680
  match &= (img1->fmt == img2->fmt);
681 682
  match &= (img1->d_w == img2->d_w);
  match &= (img1->d_h == img2->d_h);
683 684 685 686 687 688
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
  if (img1->fmt & VPX_IMG_FMT_HIGHBITDEPTH) {
    l_w *= 2;
    c_w *= 2;
  }
#endif
689

James Zern's avatar
James Zern committed
690 691 692
  for (i = 0; i < img1->d_h; ++i)
    match &= (memcmp(img1->planes[VPX_PLANE_Y] + i * img1->stride[VPX_PLANE_Y],
                     img2->planes[VPX_PLANE_Y] + i * img2->stride[VPX_PLANE_Y],
693
                     l_w) == 0);
694

James Zern's avatar
James Zern committed
695 696 697
  for (i = 0; i < c_h; ++i)
    match &= (memcmp(img1->planes[VPX_PLANE_U] + i * img1->stride[VPX_PLANE_U],
                     img2->planes[VPX_PLANE_U] + i * img2->stride[VPX_PLANE_U],
698
                     c_w) == 0);
699

James Zern's avatar
James Zern committed
700 701 702
  for (i = 0; i < c_h; ++i)
    match &= (memcmp(img1->planes[VPX_PLANE_V] + i * img1->stride[VPX_PLANE_V],
                     img2->planes[VPX_PLANE_V] + i * img2->stride[VPX_PLANE_V],
703
                     c_w) == 0);
704

John Koleszar's avatar
John Koleszar committed
705
  return match;
706 707
}

708

709
#define NELEMENTS(x) (sizeof(x)/sizeof(x[0]))
710 711
#define MAX(x,y) ((x)>(y)?(x):(y))
#if CONFIG_VP8_ENCODER && !CONFIG_VP9_ENCODER
712
#define ARG_CTRL_CNT_MAX NELEMENTS(vp8_arg_ctrl_map)
713 714 715 716 717 718
#elif !CONFIG_VP8_ENCODER && CONFIG_VP9_ENCODER
#define ARG_CTRL_CNT_MAX NELEMENTS(vp9_arg_ctrl_map)
#else
#define ARG_CTRL_CNT_MAX MAX(NELEMENTS(vp8_arg_ctrl_map), \
                             NELEMENTS(vp9_arg_ctrl_map))
#endif
John Koleszar's avatar
John Koleszar committed
719

720 721 722 723 724
#if !CONFIG_WEBM_IO
typedef int stereo_format_t;
struct EbmlGlobal { int debug; };
#endif

725
/* Per-stream configuration */
John Koleszar's avatar
John Koleszar committed
726 727 728 729
struct stream_config {
  struct vpx_codec_enc_cfg  cfg;
  const char               *out_fn;
  const char               *stats_fn;
730 731 732
#if CONFIG_FP_MB_STATS
  const char               *fpmb_stats_fn;
#endif
John Koleszar's avatar
John Koleszar committed
733 734 735 736 737
  stereo_format_t           stereo_fmt;
  int                       arg_ctrls[ARG_CTRL_CNT_MAX][2];
  int                       arg_ctrl_cnt;
  int                       write_webm;
  int                       have_kf_max_dist;
738 739 740 741
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
  // whether to use 16bit internal buffers
  int                       use_16bit_internal;
#endif
742 743 744
};


John Koleszar's avatar
John Koleszar committed
745 746 747 748 749
struct stream_state {
  int                       index;
  struct stream_state      *next;
  struct stream_config      config;
  FILE                     *file;
750
  struct rate_hist         *rate_hist;
751
  struct EbmlGlobal         ebml;
John Koleszar's avatar
John Koleszar committed
752 753 754 755 756 757 758 759 760 761
  uint64_t                  psnr_sse_total;
  uint64_t                  psnr_samples_total;
  double                    psnr_totals[4];
  int                       psnr_count;
  int                       counts[64];
  vpx_codec_ctx_t           encoder;
  unsigned int              frames_out;
  uint64_t                  cx_time;
  size_t                    nbytes;
  stats_io_t                stats;
762 763 764
#if CONFIG_FP_MB_STATS
  stats_io_t                fpmb_stats;
#endif
765
  struct vpx_image         *img;
John Koleszar's avatar
John Koleszar committed
766 767
  vpx_codec_ctx_t           decoder;
  int                       mismatch_seen;
768 769 770
};


771
void validate_positive_rational(const char          *msg,
John Koleszar's avatar
John Koleszar committed
772 773 774 775 776
                                struct vpx_rational *rat) {
  if (rat->den < 0) {
    rat->num *= -1;
    rat->den *= -1;
  }
777

John Koleszar's avatar
John Koleszar committed
778 779
  if (rat->num < 0)
    die("Error: %s must be positive\n", msg);
780

John Koleszar's avatar
John Koleszar committed
781 782
  if (!rat->den)
    die("Error: %s has zero denominator\n", msg);
783 784 785
}


786
static void parse_global_config(struct VpxEncoderConfig *global, char **argv) {
John Koleszar's avatar
John Koleszar committed
787 788 789 790 791
  char       **argi, **argj;
  struct arg   arg;

  /* Initialize default parameters */
  memset(global, 0, sizeof(*global));
792
  global->codec = get_vpx_encoder_by_index(0);
793
  global->passes = 0;
794
  global->color_type = I420;
795 796
  /* Assign default deadline to good quality */
  global->deadline = VPX_DL_GOOD_QUALITY;
John Koleszar's avatar
John Koleszar committed
797 798 799 800 801

  for (argi = argj = argv; (*argj = *argi); argi += arg.argv_step) {
    arg.argv_step = 1;

    if (arg_match(&arg, &codecarg, argi)) {
802 803 804
      global->codec = get_vpx_encoder_by_name(arg.val);
      if (!global->codec)
        die("Error: Unrecognized argument (%s) to --codec\n", arg.val);
John Koleszar's avatar
John Koleszar committed
805
    } else if (arg_match(&arg, &passes, argi)) {
John Koleszar's avatar
John Koleszar committed
806
      global->passes = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
807

John Koleszar's avatar
John Koleszar committed
808 809
      if (global->passes < 1 || global->passes > 2)
        die("Error: Invalid number of passes (%d)\n", global->passes);
John Koleszar's avatar
John Koleszar committed
810
    } else if (arg_match(&arg, &pass_arg, argi)) {
John Koleszar's avatar
John Koleszar committed
811 812 813 814 815 816 817
      global->pass = arg_parse_uint(&arg);

      if (global->pass < 1 || global->pass > 2)
        die("Error: Invalid pass selected (%d)\n",
            global->pass);
    } else if (arg_match(&arg, &usage, argi))
      global->usage = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
818
    else if (arg_match(&arg, &deadline, argi))
John Koleszar's avatar
John Koleszar committed
819
      global->deadline = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
820
    else if (arg_match(&arg, &best_dl, argi))
John Koleszar's avatar
John Koleszar committed
821
      global->deadline = VPX_DL_BEST_QUALITY;
John Koleszar's avatar
John Koleszar committed
822
    else if (arg_match(&arg, &good_dl, argi))
John Koleszar's avatar
John Koleszar committed
823
      global->deadline = VPX_DL_GOOD_QUALITY;
John Koleszar's avatar
John Koleszar committed
824
    else if (arg_match(&arg, &rt_dl, argi))
John Koleszar's avatar
John Koleszar committed
825 826
      global->deadline = VPX_DL_REALTIME;
    else if (arg_match(&arg, &use_yv12, argi))
827
      global->color_type = YV12;
John Koleszar's avatar
John Koleszar committed
828
    else if (arg_match(&arg, &use_i420, argi))
829 830 831 832 833
      global->color_type = I420;
    else if (arg_match(&arg, &use_i422, argi))
      global->color_type = I422;
    else if (arg_match(&arg, &use_i444, argi))
      global->color_type = I444;
Deb Mukherjee's avatar
Deb Mukherjee committed
834 835
    else if (arg_match(&arg, &use_i440, argi))
      global->color_type = I440;
John Koleszar's avatar
John Koleszar committed
836 837 838 839
    else if (arg_match(&arg, &quietarg, argi))
      global->quiet = 1;
    else if (arg_match(&arg, &verbosearg, argi))
      global->verbose = 1;
John Koleszar's avatar
John Koleszar committed
840
    else if (arg_match(&arg, &limit, argi))
John Koleszar's avatar
John Koleszar committed
841
      global->limit = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
842
    else if (arg_match(&arg, &skip, argi))
John Koleszar's avatar
John Koleszar committed
843
      global->skip_frames = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
844
    else if (arg_match(&arg, &psnrarg, argi))
John Koleszar's avatar
John Koleszar committed
845
      global->show_psnr = 1;
John Koleszar's avatar
John Koleszar committed
846
    else if (arg_match(&arg, &recontest, argi))
847
      global->test_decode = arg_parse_enum_or_int(&arg);
John Koleszar's avatar
John Koleszar committed
848
    else if (arg_match(&arg, &framerate, argi)) {
John Koleszar's avatar
John Koleszar committed
849 850 851 852 853
      global->framerate = arg_parse_rational(&arg);
      validate_positive_rational(arg.name, &global->framerate);
      global->have_framerate = 1;
    } else if (arg_match(&arg, &out_part, argi))
      global->out_part = 1;