diff options
author | Rodeo <[email protected]> | 2013-01-30 00:09:57 +0000 |
---|---|---|
committer | Rodeo <[email protected]> | 2013-01-30 00:09:57 +0000 |
commit | d7e26a9c04fe2a243d38002bf5a8522ccb8beedf (patch) | |
tree | 671cd5c6613ec80cf6eb69a2f1ae4878629de689 /libhb/encx264.c | |
parent | 11a7f3a63bc873c07e81583c989581152d89d3f9 (diff) |
x264 preset system: miscellaneous improvements.
See https://reviews.handbrake.fr/r/421/ for details.
git-svn-id: svn://svn.handbrake.fr/HandBrake/trunk@5225 b64f7644-9d1e-0410-96f1-a4d463321fa5
Diffstat (limited to 'libhb/encx264.c')
-rw-r--r-- | libhb/encx264.c | 198 |
1 files changed, 122 insertions, 76 deletions
diff --git a/libhb/encx264.c b/libhb/encx264.c index 503f46cd9..b52832416 100644 --- a/libhb/encx264.c +++ b/libhb/encx264.c @@ -275,9 +275,9 @@ int encx264Init( hb_work_object_t * w, hb_job_t * job ) } /* Apply profile and level settings last, if present. */ - if (job->x264_profile != NULL && *job->x264_profile) + if (job->h264_profile != NULL && *job->h264_profile) { - if (x264_param_apply_profile(¶m, job->x264_profile)) + if (hb_apply_h264_profile(¶m, job->h264_profile, 1)) { free(pv); pv = NULL; @@ -287,7 +287,7 @@ int encx264Init( hb_work_object_t * w, hb_job_t * job ) if (job->h264_level != NULL && *job->h264_level) { if (hb_apply_h264_level(¶m, job->h264_level, - job->x264_profile, 1) < 0) + job->h264_profile, 1) < 0) { free(pv); pv = NULL; @@ -629,6 +629,60 @@ int encx264Work( hb_work_object_t * w, hb_buffer_t ** buf_in, return HB_WORK_OK; } +int hb_apply_h264_profile(x264_param_t *param, const char *h264_profile, + int verbose) +{ + if (h264_profile != NULL && + strcasecmp(h264_profile, hb_h264_profile_names[0]) != 0) + { + /* + * baseline profile doesn't support interlacing + */ + if ((param->b_interlaced || + param->b_fake_interlaced) && + !strcasecmp(h264_profile, "baseline")) + { + if (verbose) + { + hb_log("hb_apply_h264_profile [warning]: baseline profile doesn't support interlacing, disabling"); + } + param->b_interlaced = param->b_fake_interlaced = 0; + } + /* + * lossless requires High 4:4:4 Predictive profile + */ + if (param->rc.f_rf_constant < 1.0 && + param->rc.i_rc_method == X264_RC_CRF && + strcasecmp(h264_profile, "high444") != 0) + { + if (verbose) + { + hb_log("hb_apply_h264_profile [warning]: lossless requires high444 profile, disabling"); + } + param->rc.f_rf_constant = 1.0; + } + if (!strcasecmp(h264_profile, "high10") || + !strcasecmp(h264_profile, "high422")) + { + // arbitrary profile names may be specified via the CLI + // map unsupported high10 and high422 profiles to high + return x264_param_apply_profile(param, "high"); + } + return x264_param_apply_profile(param, h264_profile); + } + else if (!strcasecmp(h264_profile, hb_h264_profile_names[0])) + { + // "auto", do nothing + return 0; + } + else + { + // error (profile not a string), abort + hb_error("hb_apply_h264_profile: no profile specified"); + return -1; + } +} + int hb_check_h264_level(const char *h264_level, int width, int height, int fps_num, int fps_den, int interlaced, int fake_interlaced) @@ -645,24 +699,65 @@ int hb_check_h264_level(const char *h264_level, int width, int height, } int hb_apply_h264_level(x264_param_t *param, const char *h264_level, - const char *x264_profile, int verbose) + const char *h264_profile, int verbose) { float f_framerate; const x264_level_t *x264_level = NULL; int i, i_mb_size, i_mb_rate, i_mb_width, i_mb_height, max_mb_side, ret; /* + * find the x264_level_t corresponding to the requested level + */ + if (h264_level != NULL && + strcasecmp(h264_level, hb_h264_level_names[0]) != 0) + { + for (i = 0; hb_h264_level_values[i]; i++) + { + if (!strcmp(hb_h264_level_names[i], h264_level)) + { + int val = hb_h264_level_values[i]; + for (i = 0; x264_levels[i].level_idc; i++) + { + if (x264_levels[i].level_idc == val) + { + x264_level = &x264_levels[i]; + break; + } + } + break; + } + } + if (x264_level == NULL) + { + // error (invalid or unsupported level), abort + hb_error("hb_apply_h264_level: invalid level %s", h264_level); + return -1; + } + } + else if(!strcasecmp(h264_level, hb_h264_level_names[0])) + { + // "auto", do nothing + return 0; + } + else + { + // error (level not a string), abort + hb_error("hb_apply_h264_level: no level specified"); + return -1; + } + + /* * the H.264 profile determines VBV constraints */ enum { // Main or Baseline (equivalent) - HB_H264_PROFILE_MAIN, + HB_ENCX264_PROFILE_MAIN, // High (no 4:2:2 or 10-bit support, so anything lossy is equivalent) - HB_H264_PROFILE_HIGH, + HB_ENCX264_PROFILE_HIGH, // Lossless (4:2:0 8-bit for now) - HB_H264_PROFILE_HIGH444, - } h264_profile; + HB_ENCX264_PROFILE_HIGH444, + } hb_encx264_profile; /* * H.264 profile @@ -674,21 +769,21 @@ int hb_apply_h264_level(x264_param_t *param, const char *h264_level, * re-enabled. */ #if 0 - if (x264_profile != NULL && *x264_profile) + if (h264_profile != NULL && *h264_profile) { // if the user explicitly specified a profile, don't guess it - if (!strcasecmp(x264_profile, "high444")) + if (!strcasecmp(h264_profile, "high444")) { - h264_profile = HB_H264_PROFILE_HIGH444; + hb_encx264_profile = HB_ENCX264_PROFILE_HIGH444; } - else if (!strcasecmp(x264_profile, "main") || - !strcasecmp(x264_profile, "baseline")) + else if (!strcasecmp(h264_profile, "main") || + !strcasecmp(h264_profile, "baseline")) { - h264_profile = HB_H264_PROFILE_MAIN; + hb_encx264_profile = HB_ENCX264_PROFILE_MAIN; } else { - h264_profile = HB_H264_PROFILE_HIGH; + hb_encx264_profile = HB_ENCX264_PROFILE_HIGH; } } else @@ -698,53 +793,18 @@ int hb_apply_h264_level(x264_param_t *param, const char *h264_level, if (param->rc.i_rc_method == X264_RC_CRF && param->rc.f_rf_constant < 1.0) { - h264_profile = HB_H264_PROFILE_HIGH444; + hb_encx264_profile = HB_ENCX264_PROFILE_HIGH444; } else if (param->analyse.b_transform_8x8 || param->i_cqm_preset != X264_CQM_FLAT) { - h264_profile = HB_H264_PROFILE_HIGH; + hb_encx264_profile = HB_ENCX264_PROFILE_HIGH; } else { - h264_profile = HB_H264_PROFILE_MAIN; - } - } - - /* - * find the x264_level_t corresponding to the requested level - */ - if (h264_level != NULL && *h264_level) - { - for (i = 0; hb_h264_level_values[i]; i++) - { - if (!strcmp(hb_h264_level_names[i], h264_level)) - { - int val = hb_h264_level_values[i]; - for (i = 0; x264_levels[i].level_idc; i++) - { - if (x264_levels[i].level_idc == val) - { - x264_level = &x264_levels[i]; - break; - } - } - break; - } - } - if (x264_level == NULL) - { - // error (invalid or unsupported level), abort - hb_error("hb_apply_h264_level: invalid level %s", h264_level); - return -1; + hb_encx264_profile = HB_ENCX264_PROFILE_MAIN; } } - else - { - // error (level not a string), abort - hb_error("hb_apply_h264_level: no level specified"); - return -1; - } /* * we need at least width and height in order to apply a level correctly @@ -825,10 +885,10 @@ int hb_apply_h264_level(x264_param_t *param, const char *h264_level, /* * set and/or sanitize the VBV (if not lossless) */ - if (h264_profile != HB_H264_PROFILE_HIGH444) + if (hb_encx264_profile != HB_ENCX264_PROFILE_HIGH444) { // High profile allows for higher VBV bufsize/maxrate - int cbp_factor = h264_profile == HB_H264_PROFILE_HIGH ? 5 : 4; + int cbp_factor = hb_encx264_profile == HB_ENCX264_PROFILE_HIGH ? 5 : 4; if (!param->rc.i_vbv_max_bitrate) { param->rc.i_vbv_max_bitrate = (x264_level->bitrate * cbp_factor) / 4; @@ -918,7 +978,7 @@ int hb_apply_h264_level(x264_param_t *param, const char *h264_level, } char * hb_x264_param_unparse(const char *x264_preset, const char *x264_tune, - const char *x264_encopts, const char *x264_profile, + const char *x264_encopts, const char *h264_profile, const char *h264_level, int width, int height) { int i; @@ -979,24 +1039,10 @@ char * hb_x264_param_unparse(const char *x264_preset, const char *x264_tune, /* * apply the x264 profile, if specified */ - if (x264_profile != NULL && *x264_profile) + if (h264_profile != NULL && *h264_profile) { - if (x264_param_apply_profile(¶m, x264_profile) < 0) - { - /* - * Note: x264_param_apply_profile can bail even when the specified - * profile is valid; examples: - * - * 1) when profile is baseline and interlaced is set - * 2) when profile is not high444 and lossless is set - * - * So this can happen even when calling hb_x264_param_unparse from a - * GUI - maybe we should sanitize settings before calling - * x264_param_apply_profile()? - */ - hb_dict_free(&x264_opts); - return strdup("hb_x264_param_unparse: could not apply x264 profile"); - } + // be quiet so at to not pollute GUI logs + hb_apply_h264_profile(¶m, h264_profile, 0); } /* @@ -1008,7 +1054,7 @@ char * hb_x264_param_unparse(const char *x264_preset, const char *x264_tune, param.i_width = width; param.i_height = height; // be quiet so at to not pollute GUI logs - hb_apply_h264_level(¶m, h264_level, x264_profile, 0); + hb_apply_h264_level(¶m, h264_level, h264_profile, 0); } /* @@ -1590,9 +1636,9 @@ const char * const * hb_x264_tunes() return x264_tune_names; } -const char * const * hb_x264_profiles() +const char * const * hb_h264_profiles() { - return x264_profile_names; + return hb_h264_profile_names; } const char * const * hb_h264_levels() |