# HG changeset patch # User tack # Date 1276206320 0 # Node ID 97c3077b4e6782524c6d66ff1b36fbd2d64f9c2e # Parent c1d967987dc2736acf96cf8bfb4f720cb6798f8c ve_x264: rewrite option parsing and add support for x264's presets, profiles, and tuning system. No longer explicitly disable 'psnr' and 'ssim' options (they are off by default); deprecate the 'turbo' option, enable fast first pass by default, and introduce slow_firstpass option to disable it (provides parity with x264). Some ideas taken from patch submitted by Micha (mk spline de). diff -r c1d967987dc2 -r 97c3077b4e67 libmpcodecs/ve_x264.c --- a/libmpcodecs/ve_x264.c Thu Jun 10 16:01:59 2010 +0000 +++ b/libmpcodecs/ve_x264.c Thu Jun 10 21:45:20 2010 +0000 @@ -56,7 +56,6 @@ } h264_module_t; extern char* passtmpfile; -static int turbo = 0; static x264_param_t param; static int parse_error = 0; @@ -65,74 +64,78 @@ void x264enc_set_param(const m_option_t* opt, char* arg) { - static int initted = 0; - if(!initted) { + static int initialized = 0; + int slow_firstpass = 0; + char *preset = NULL, *tune = NULL, *profile = NULL; + char *p, *copy, *name; + + if (!initialized) { x264_param_default(¶m); - x264_param_parse(¶m, "psnr", "no"); - x264_param_parse(¶m, "ssim", "no"); - initted = 1; + initialized = 1; } - - if(!arg) { + if (!arg) { + mp_msg(MSGT_CFGPARSER, MSGL_ERR, "Option x264encopts: no options provided\n"); parse_error = 1; return; - } - - while(*arg) { - char *name = arg; - char *value; - int ret; + } else if (!*arg) + /* Empty arguments, just doing initialization of default parameters. */ + return; - arg += strcspn(arg, ":"); - if(*arg) { - *arg = 0; - arg++; - } + /* Step 1: look for initial preset/tune. */ + copy = p = strdup(arg); + while ((name = strsep(©, ":"))) { + char *value = strpbrk(name, "=:"); + if (!value) + continue; + *value++ = 0; + if (!strcasecmp(name, "preset")) + preset = value; + else if (!strcasecmp(name, "tune")) + tune = value; + } + if (x264_param_default_preset(¶m, preset, tune) < 0) { + mp_msg(MSGT_CFGPARSER, MSGL_ERR, "Option x264encopts: Invalid preset or tune.\n"); + parse_error = 1; + } + free(p); - value = strchr( name, '=' ); - if(value) { - *value = 0; - value++; - } + /* Step 2: explicit user overrides */ + while ((name = strsep(&arg, ":")) && *name) { + int ret = 0; + char *value = strpbrk(name, "=:"); - if(!strcmp(name, "turbo")) { - turbo = value ? atoi(value) : 1; - continue; - } - + if (value) + *value++ = 0; + if (!strcasecmp(name, "profile")) + profile = value; + else if (!strcasecmp(name, "turbo")) { + mp_msg(MSGT_CFGPARSER, MSGL_WARN, "Option x264encopts: turbo option is deprecated; " + "use slow_firstpass to disable turbo\n"); + if (value && *value == '0') + slow_firstpass = 1; + } else if (!strcasecmp(name, "slow_firstpass")) + slow_firstpass = 1; + else if (strcasecmp(name, "preset") && strcasecmp(name, "tune")) { ret = x264_param_parse(¶m, name, value); if(ret == X264_PARAM_BAD_NAME) mp_msg(MSGT_CFGPARSER, MSGL_ERR, "Option x264encopts: Unknown suboption %s\n", name); if(ret == X264_PARAM_BAD_VALUE) mp_msg(MSGT_CFGPARSER, MSGL_ERR, "Option x264encopts: Bad argument %s=%s\n", name, value ? value : "(null)"); + } /* mark this option as done, so it's not reparsed if there's another -x264encopts */ *name = 0; parse_error |= ret; } - if(param.rc.b_stat_write && !param.rc.b_stat_read) { - /* Adjust or disable some flags to gain speed in the first pass */ - if(turbo == 1) - { - param.i_frame_reference = ( param.i_frame_reference + 1 ) >> 1; - param.analyse.i_subpel_refine = FFMAX( FFMIN( 3, param.analyse.i_subpel_refine - 1 ), 1 ); - param.analyse.inter &= ( ~X264_ANALYSE_PSUB8x8 ); - param.analyse.inter &= ( ~X264_ANALYSE_BSUB16x16 ); - param.analyse.i_trellis = 0; - } - else if(turbo >= 2) - { - param.i_frame_reference = 1; - param.analyse.i_subpel_refine = 1; - param.analyse.i_me_method = X264_ME_DIA; - param.analyse.inter = 0; - param.analyse.b_transform_8x8 = 0; - param.analyse.b_weighted_bipred = 0; - param.analyse.i_trellis = 0; - } - } + /* Step 3: Apply fast first pass (turbo) options. */ + if (!slow_firstpass) + x264_param_apply_fastfirstpass(¶m); + + /* Step 4: enforce profile */ + if (profile && x264_param_apply_profile(¶m, profile) < 0) + parse_error = 1; } static int config(struct vf_instance *vf, int width, int height, int d_width, int d_height, unsigned int flags, unsigned int outfmt) {