vpxenc.c 82.4 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");
John Koleszar's avatar
John Koleszar committed
143
static const arg_def_t codecarg = ARG_DEF(NULL, "codec", 1,
John Koleszar's avatar
John Koleszar committed
144
                                          "Codec to use");
John Koleszar's avatar
John Koleszar committed
145
static const arg_def_t passes           = ARG_DEF("p", "passes", 1,
John Koleszar's avatar
John Koleszar committed
146
                                                  "Number of passes (1/2)");
John Koleszar's avatar
John Koleszar committed
147
static const arg_def_t pass_arg         = ARG_DEF(NULL, "pass", 1,
John Koleszar's avatar
John Koleszar committed
148
                                                  "Pass to execute (1/2)");
John Koleszar's avatar
John Koleszar committed
149
static const arg_def_t fpf_name         = ARG_DEF(NULL, "fpf", 1,
John Koleszar's avatar
John Koleszar committed
150
                                                  "First pass statistics file name");
151 152 153 154
#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
155 156
static const arg_def_t limit = ARG_DEF(NULL, "limit", 1,
                                       "Stop encoding after n input frames");
157
static const arg_def_t skip = ARG_DEF(NULL, "skip", 1,
John Koleszar's avatar
John Koleszar committed
158
                                      "Skip the first n input frames");
John Koleszar's avatar
John Koleszar committed
159
static const arg_def_t deadline         = ARG_DEF("d", "deadline", 1,
John Koleszar's avatar
John Koleszar committed
160
                                                  "Deadline per frame (usec)");
John Koleszar's avatar
John Koleszar committed
161
static const arg_def_t best_dl          = ARG_DEF(NULL, "best", 0,
John Koleszar's avatar
John Koleszar committed
162
                                                  "Use Best Quality Deadline");
John Koleszar's avatar
John Koleszar committed
163
static const arg_def_t good_dl          = ARG_DEF(NULL, "good", 0,
John Koleszar's avatar
John Koleszar committed
164
                                                  "Use Good Quality Deadline");
John Koleszar's avatar
John Koleszar committed
165
static const arg_def_t rt_dl            = ARG_DEF(NULL, "rt", 0,
John Koleszar's avatar
John Koleszar committed
166
                                                  "Use Realtime Quality Deadline");
James Zern's avatar
James Zern committed
167
static const arg_def_t quietarg         = ARG_DEF("q", "quiet", 0,
John Koleszar's avatar
John Koleszar committed
168
                                                  "Do not print encode progress");
John Koleszar's avatar
John Koleszar committed
169
static const arg_def_t verbosearg       = ARG_DEF("v", "verbose", 0,
John Koleszar's avatar
John Koleszar committed
170
                                                  "Show encoder parameters");
John Koleszar's avatar
John Koleszar committed
171
static const arg_def_t psnrarg          = ARG_DEF(NULL, "psnr", 0,
John Koleszar's avatar
John Koleszar committed
172
                                                  "Show PSNR in status line");
173

174 175 176 177 178 179 180 181 182
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);
183
static const arg_def_t framerate        = ARG_DEF(NULL, "fps", 1,
John Koleszar's avatar
John Koleszar committed
184
                                                  "Stream frame rate (rate/scale)");
John Koleszar's avatar
John Koleszar committed
185
static const arg_def_t use_ivf          = ARG_DEF(NULL, "ivf", 0,
186
                                                  "Output IVF (default is WebM if WebM IO is enabled)");
187
static const arg_def_t out_part = ARG_DEF("P", "output-partitions", 0,
John Koleszar's avatar
John Koleszar committed
188
                                          "Makes encoder output partitions. Requires IVF output!");
189
static const arg_def_t q_hist_n         = ARG_DEF(NULL, "q-hist", 1,
John Koleszar's avatar
John Koleszar committed
190
                                                  "Show quantizer histogram (n-buckets)");
191
static const arg_def_t rate_hist_n         = ARG_DEF(NULL, "rate-hist", 1,
John Koleszar's avatar
John Koleszar committed
192
                                                     "Show rate histogram (n-buckets)");
193 194 195
static const arg_def_t disable_warnings =
    ARG_DEF(NULL, "disable-warnings", 0,
            "Disable warnings about potentially incorrect encode settings.");
196 197 198
static const arg_def_t disable_warning_prompt =
    ARG_DEF("y", "disable-warning-prompt", 0,
            "Display warnings, but do not prompt user to continue.");
199 200 201 202
static const arg_def_t experimental_bitstream =
    ARG_DEF(NULL, "experimental-bitstream", 0,
            "Allow experimental bitstream features.");

203 204 205 206
#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
207

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

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

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

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


static const arg_def_t bias_pct = ARG_DEF(NULL, "bias-pct", 1,
John Koleszar's avatar
John Koleszar committed
307
                                          "CBR/VBR bias (0=CBR, 100=VBR)");
John Koleszar's avatar
John Koleszar committed
308
static const arg_def_t minsection_pct = ARG_DEF(NULL, "minsection-pct", 1,
John Koleszar's avatar
John Koleszar committed
309
                                                "GOP min bitrate (% of target)");
John Koleszar's avatar
John Koleszar committed
310
static const arg_def_t maxsection_pct = ARG_DEF(NULL, "maxsection-pct", 1,
John Koleszar's avatar
John Koleszar committed
311 312 313
                                                "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
314 315 316 317
};


static const arg_def_t kf_min_dist = ARG_DEF(NULL, "kf-min-dist", 1,
John Koleszar's avatar
John Koleszar committed
318
                                             "Minimum keyframe interval (frames)");
John Koleszar's avatar
John Koleszar committed
319
static const arg_def_t kf_max_dist = ARG_DEF(NULL, "kf-max-dist", 1,
John Koleszar's avatar
John Koleszar committed
320
                                             "Maximum keyframe interval (frames)");
321
static const arg_def_t kf_disabled = ARG_DEF(NULL, "disable-kf", 0,
John Koleszar's avatar
John Koleszar committed
322 323 324
                                             "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
325 326 327 328
};


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

356
#if CONFIG_VP8_ENCODER
357 358
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
359 360 361
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
362
  &tune_ssim, &cq_level, &max_intra_rate_pct,
363
  NULL
John Koleszar's avatar
John Koleszar committed
364
};
365 366 367 368 369 370 371 372 373 374 375
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
376 377 378 379 380 381 382 383 384
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,
385
    "Adaptive quantization mode (0: off (default), 1: variance 2: complexity, "
386 387
    "3: cyclic refresh)");
static const arg_def_t frame_periodic_boost = ARG_DEF(
388
    NULL, "frame-boost", 1,
389
    "Enable frame periodic boost (0: off (default), 1: on)");
390

391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407
#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

408 409 410 411 412 413 414 415 416
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);

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

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

441
void usage_exit() {
John Koleszar's avatar
John Koleszar committed
442 443 444 445 446 447
  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
448
  arg_show_usage(stderr, main_args);
John Koleszar's avatar
John Koleszar committed
449
  fprintf(stderr, "\nEncoder Global Options:\n");
John Koleszar's avatar
John Koleszar committed
450
  arg_show_usage(stderr, global_args);
John Koleszar's avatar
John Koleszar committed
451
  fprintf(stderr, "\nRate Control Options:\n");
John Koleszar's avatar
John Koleszar committed
452
  arg_show_usage(stderr, rc_args);
John Koleszar's avatar
John Koleszar committed
453
  fprintf(stderr, "\nTwopass Rate Control Options:\n");
John Koleszar's avatar
John Koleszar committed
454
  arg_show_usage(stderr, rc_twopass_args);
John Koleszar's avatar
John Koleszar committed
455
  fprintf(stderr, "\nKeyframe Placement Options:\n");
John Koleszar's avatar
John Koleszar committed
456
  arg_show_usage(stderr, kf_args);
John Koleszar's avatar
John Koleszar committed
457
#if CONFIG_VP8_ENCODER
John Koleszar's avatar
John Koleszar committed
458
  fprintf(stderr, "\nVP8 Specific Options:\n");
John Koleszar's avatar
John Koleszar committed
459
  arg_show_usage(stderr, vp8_args);
John Koleszar's avatar
John Koleszar committed
460
#endif
461 462
#if CONFIG_VP9_ENCODER
  fprintf(stderr, "\nVP9 Specific Options:\n");
John Koleszar's avatar
John Koleszar committed
463
  arg_show_usage(stderr, vp9_args);
John Koleszar's avatar
John Koleszar committed
464
#endif
John Koleszar's avatar
John Koleszar committed
465 466 467
  fprintf(stderr, "\nStream timebase (--timebase):\n"
          "  The desired precision of timestamps in the output, expressed\n"
          "  in fractional seconds. Default is 1/1000.\n");
468
  fprintf(stderr, "\nIncluded encoders:\n\n");
John Koleszar's avatar
John Koleszar committed
469

470 471
  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
472
    fprintf(stderr, "    %-6s - %s\n",
473
            encoder->name, vpx_codec_iface_name(encoder->codec_interface()));
474
  }
John Koleszar's avatar
John Koleszar committed
475 476

  exit(EXIT_FAILURE);
John Koleszar's avatar
John Koleszar committed
477 478
}

479
#define mmin(a, b)  ((a) < (b) ? (a) : (b))
480 481 482 483 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

#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
576 577
static void find_mismatch(const vpx_image_t *const img1,
                          const vpx_image_t *const img2,
578
                          int yloc[4], int uloc[4], int vloc[4]) {
579 580 581 582 583 584 585 586 587
  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;
588
  yloc[0] = yloc[1] = yloc[2] = yloc[3] = -1;
589 590
  for (i = 0, match = 1; match && i < img1->d_h; i += bsize) {
    for (j = 0; match && j < img1->d_w; j += bsize) {
591
      int k, l;
James Zern's avatar
James Zern committed
592 593 594 595
      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) {
596 597 598 599 600 601
          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;
602 603 604 605
            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);
606
            match = 0;
607
            break;
608 609
          }
        }
James Zern's avatar
James Zern committed
610
      }
611
    }
612
  }
613

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

James Zern's avatar
James Zern committed
666 667
static int compare_img(const vpx_image_t *const img1,
                       const vpx_image_t *const img2) {
668 669
  uint32_t l_w = img1->d_w;
  uint32_t c_w =
670 671 672 673
      (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
674
  int match = 1;
675

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

James Zern's avatar
James Zern committed
686 687 688
  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],
689
                     l_w) == 0);
690

James Zern's avatar
James Zern committed
691 692 693
  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],
694
                     c_w) == 0);
695

James Zern's avatar
James Zern committed
696 697 698
  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],
699
                     c_w) == 0);
700

John Koleszar's avatar
John Koleszar committed
701
  return match;
702 703
}

704

705
#define NELEMENTS(x) (sizeof(x)/sizeof(x[0]))
706 707
#define MAX(x,y) ((x)>(y)?(x):(y))
#if CONFIG_VP8_ENCODER && !CONFIG_VP9_ENCODER
708
#define ARG_CTRL_CNT_MAX NELEMENTS(vp8_arg_ctrl_map)
709 710 711 712 713 714
#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
715

716 717 718 719 720
#if !CONFIG_WEBM_IO
typedef int stereo_format_t;
struct EbmlGlobal { int debug; };
#endif

721
/* Per-stream configuration */
John Koleszar's avatar
John Koleszar committed
722 723 724 725
struct stream_config {
  struct vpx_codec_enc_cfg  cfg;
  const char               *out_fn;
  const char               *stats_fn;
726 727 728
#if CONFIG_FP_MB_STATS
  const char               *fpmb_stats_fn;
#endif
John Koleszar's avatar
John Koleszar committed
729 730 731 732 733
  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;
734 735 736 737
#if CONFIG_VP9 && CONFIG_VP9_HIGHBITDEPTH
  // whether to use 16bit internal buffers
  int                       use_16bit_internal;
#endif
738 739 740
};


John Koleszar's avatar
John Koleszar committed
741 742 743 744 745
struct stream_state {
  int                       index;
  struct stream_state      *next;
  struct stream_config      config;
  FILE                     *file;
746
  struct rate_hist         *rate_hist;
747
  struct EbmlGlobal         ebml;
John Koleszar's avatar
John Koleszar committed
748 749 750 751 752 753 754 755 756 757
  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;
758 759 760
#if CONFIG_FP_MB_STATS
  stats_io_t                fpmb_stats;
#endif
761
  struct vpx_image         *img;
John Koleszar's avatar
John Koleszar committed
762 763
  vpx_codec_ctx_t           decoder;
  int                       mismatch_seen;
764 765 766
};


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

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

John Koleszar's avatar
John Koleszar committed
777 778
  if (!rat->den)
    die("Error: %s has zero denominator\n", msg);
779 780 781
}


782
static void parse_global_config(struct VpxEncoderConfig *global, char **argv) {
John Koleszar's avatar
John Koleszar committed
783 784 785 786 787
  char       **argi, **argj;
  struct arg   arg;

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

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

    if (arg_match(&arg, &codecarg, argi)) {
798 799 800
      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
801
    } else if (arg_match(&arg, &passes, argi)) {
John Koleszar's avatar
John Koleszar committed
802
      global->passes = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
803

John Koleszar's avatar
John Koleszar committed
804 805
      if (global->passes < 1 || global->passes > 2)
        die("Error: Invalid number of passes (%d)\n", global->passes);
John Koleszar's avatar
John Koleszar committed
806
    } else if (arg_match(&arg, &pass_arg, argi)) {
John Koleszar's avatar
John Koleszar committed
807 808 809 810 811 812 813
      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
814
    else if (arg_match(&arg, &deadline, argi))
John Koleszar's avatar
John Koleszar committed
815
      global->deadline = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
816
    else if (arg_match(&arg, &best_dl, argi))
John Koleszar's avatar
John Koleszar committed
817
      global->deadline = VPX_DL_BEST_QUALITY;
John Koleszar's avatar
John Koleszar committed
818
    else if (arg_match(&arg, &good_dl, argi))
John Koleszar's avatar
John Koleszar committed
819
      global->deadline = VPX_DL_GOOD_QUALITY;
John Koleszar's avatar
John Koleszar committed
820
    else if (arg_match(&arg, &rt_dl, argi))
John Koleszar's avatar
John Koleszar committed
821 822
      global->deadline = VPX_DL_REALTIME;
    else if (arg_match(&arg, &use_yv12, argi))
823
      global->color_type = YV12;
John Koleszar's avatar
John Koleszar committed
824
    else if (arg_match(&arg, &use_i420, argi))
825 826 827 828 829
      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;
John Koleszar's avatar
John Koleszar committed
830 831 832 833
    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
834
    else if (arg_match(&arg, &limit, argi))
John Koleszar's avatar
John Koleszar committed
835
      global->limit = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
836
    else if (arg_match(&arg, &skip, argi))
John Koleszar's avatar
John Koleszar committed
837
      global->skip_frames = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
838
    else if (arg_match(&arg, &psnrarg, argi))
John Koleszar's avatar
John Koleszar committed
839
      global->show_psnr = 1;
John Koleszar's avatar
John Koleszar committed
840
    else if (arg_match(&arg, &recontest, argi))
841
      global->test_decode = arg_parse_enum_or_int(&arg);
John Koleszar's avatar
John Koleszar committed
842
    else if (arg_match(&arg, &framerate, argi)) {
John Koleszar's avatar
John Koleszar committed
843 844 845 846 847
      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;
John Koleszar's avatar
John Koleszar committed
848
    else if (arg_match(&arg, &debugmode, argi))
John Koleszar's avatar
John Koleszar committed
849
      global->debug = 1;
John Koleszar's avatar
John Koleszar committed
850
    else if (arg_match(&arg, &q_hist_n, argi))
John Koleszar's avatar
John Koleszar committed
851
      global->show_q_hist_buckets = arg_parse_uint(&arg);
John Koleszar's avatar
John Koleszar committed
852
    else if (arg_match(&arg, &rate_hist_n, argi))