config: handle NULL value when parsing non-bools

When the config parser sees an "implicit" bool like:

  [core]
  someVariable

it passes NULL to the config callback. Any callback code which expects a
string must check for NULL. This usually happens via helpers like
git_config_string(), etc, but some custom code forgets to do so and will
segfault.

These are all fairly vanilla cases where the solution is just the usual
pattern of:

  if (!value)
        return config_error_nonbool(var);

though note that in a few cases we have to split initializers like:

  int some_var = initializer();

into:

  int some_var;
  if (!value)
        return config_error_nonbool(var);
  some_var = initializer();

There are still some broken instances after this patch, which I'll
address on their own in individual patches after this one.

Reported-by: Carlos Andrés Ramírez Cataño <antaigroupltda@gmail.com>
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
maint
Jeff King 2023-12-07 02:11:14 -05:00 committed by Junio C Hamano
parent 564d0252ca
commit ba176db511
11 changed files with 47 additions and 5 deletions

View File

@ -748,6 +748,8 @@ static int git_blame_config(const char *var, const char *value,
} }


if (!strcmp(var, "blame.coloring")) { if (!strcmp(var, "blame.coloring")) {
if (!value)
return config_error_nonbool(var);
if (!strcmp(value, "repeatedLines")) { if (!strcmp(value, "repeatedLines")) {
coloring_mode |= OUTPUT_COLOR_LINE; coloring_mode |= OUTPUT_COLOR_LINE;
} else if (!strcmp(value, "highlightRecent")) { } else if (!strcmp(value, "highlightRecent")) {

View File

@ -1202,6 +1202,8 @@ static int git_checkout_config(const char *var, const char *value,
struct checkout_opts *opts = cb; struct checkout_opts *opts = cb;


if (!strcmp(var, "diff.ignoresubmodules")) { if (!strcmp(var, "diff.ignoresubmodules")) {
if (!value)
return config_error_nonbool(var);
handle_ignore_submodules_arg(&opts->diff_options, value); handle_ignore_submodules_arg(&opts->diff_options, value);
return 0; return 0;
} }

View File

@ -791,6 +791,8 @@ static int git_clone_config(const char *k, const char *v,
const struct config_context *ctx, void *cb) const struct config_context *ctx, void *cb)
{ {
if (!strcmp(k, "clone.defaultremotename")) { if (!strcmp(k, "clone.defaultremotename")) {
if (!v)
return config_error_nonbool(k);
free(remote_name); free(remote_name);
remote_name = xstrdup(v); remote_name = xstrdup(v);
} }

View File

@ -594,8 +594,11 @@ static int git_log_config(const char *var, const char *value,
decoration_style = 0; /* maybe warn? */ decoration_style = 0; /* maybe warn? */
return 0; return 0;
} }
if (!strcmp(var, "log.diffmerges")) if (!strcmp(var, "log.diffmerges")) {
if (!value)
return config_error_nonbool(var);
return diff_merges_config(value); return diff_merges_config(value);
}
if (!strcmp(var, "log.showroot")) { if (!strcmp(var, "log.showroot")) {
default_show_root = git_config_bool(var, value); default_show_root = git_config_bool(var, value);
return 0; return 0;

View File

@ -3204,7 +3204,7 @@ static int git_pack_config(const char *k, const char *v,
return 0; return 0;
} }
if (!strcmp(k, "uploadpack.blobpackfileuri")) { if (!strcmp(k, "uploadpack.blobpackfileuri")) {
struct configured_exclusion *ex = xmalloc(sizeof(*ex)); struct configured_exclusion *ex;
const char *oid_end, *pack_end; const char *oid_end, *pack_end;
/* /*
* Stores the pack hash. This is not a true object ID, but is * Stores the pack hash. This is not a true object ID, but is
@ -3212,6 +3212,10 @@ static int git_pack_config(const char *k, const char *v,
*/ */
struct object_id pack_hash; struct object_id pack_hash;


if (!v)
return config_error_nonbool(k);

ex = xmalloc(sizeof(*ex));
if (parse_oid_hex(v, &ex->e.oid, &oid_end) || if (parse_oid_hex(v, &ex->e.oid, &oid_end) ||
*oid_end != ' ' || *oid_end != ' ' ||
parse_oid_hex(oid_end + 1, &pack_hash, &pack_end) || parse_oid_hex(oid_end + 1, &pack_hash, &pack_end) ||

View File

@ -255,6 +255,8 @@ int mingw_core_config(const char *var, const char *value,
} }


if (!strcmp(var, "core.unsetenvvars")) { if (!strcmp(var, "core.unsetenvvars")) {
if (!value)
return config_error_nonbool(var);
free(unset_environment_variables); free(unset_environment_variables);
unset_environment_variables = xstrdup(value); unset_environment_variables = xstrdup(value);
return 0; return 0;

View File

@ -1386,6 +1386,8 @@ static int git_default_core_config(const char *var, const char *value,
return 0; return 0;
} }
if (!strcmp(var, "core.checkstat")) { if (!strcmp(var, "core.checkstat")) {
if (!value)
return config_error_nonbool(var);
if (!strcasecmp(value, "default")) if (!strcasecmp(value, "default"))
check_stat = 1; check_stat = 1;
else if (!strcasecmp(value, "minimal")) else if (!strcasecmp(value, "minimal"))
@ -1547,11 +1549,15 @@ static int git_default_core_config(const char *var, const char *value,
} }


if (!strcmp(var, "core.checkroundtripencoding")) { if (!strcmp(var, "core.checkroundtripencoding")) {
if (!value)
return config_error_nonbool(var);
check_roundtrip_encoding = xstrdup(value); check_roundtrip_encoding = xstrdup(value);
return 0; return 0;
} }


if (!strcmp(var, "core.notesref")) { if (!strcmp(var, "core.notesref")) {
if (!value)
return config_error_nonbool(var);
notes_ref_name = xstrdup(value); notes_ref_name = xstrdup(value);
return 0; return 0;
} }
@ -1619,6 +1625,8 @@ static int git_default_core_config(const char *var, const char *value,
} }


if (!strcmp(var, "core.createobject")) { if (!strcmp(var, "core.createobject")) {
if (!value)
return config_error_nonbool(var);
if (!strcmp(value, "rename")) if (!strcmp(value, "rename"))
object_creation_mode = OBJECT_CREATION_USES_RENAMES; object_creation_mode = OBJECT_CREATION_USES_RENAMES;
else if (!strcmp(value, "link")) else if (!strcmp(value, "link"))

19
diff.c
View File

@ -372,7 +372,10 @@ int git_diff_ui_config(const char *var, const char *value,
return 0; return 0;
} }
if (!strcmp(var, "diff.colormovedws")) { if (!strcmp(var, "diff.colormovedws")) {
unsigned cm = parse_color_moved_ws(value); unsigned cm;
if (!value)
return config_error_nonbool(var);
cm = parse_color_moved_ws(value);
if (cm & COLOR_MOVED_WS_ERROR) if (cm & COLOR_MOVED_WS_ERROR)
return -1; return -1;
diff_color_moved_ws_default = cm; diff_color_moved_ws_default = cm;
@ -426,10 +429,15 @@ int git_diff_ui_config(const char *var, const char *value,
if (!strcmp(var, "diff.orderfile")) if (!strcmp(var, "diff.orderfile"))
return git_config_pathname(&diff_order_file_cfg, var, value); return git_config_pathname(&diff_order_file_cfg, var, value);


if (!strcmp(var, "diff.ignoresubmodules")) if (!strcmp(var, "diff.ignoresubmodules")) {
if (!value)
return config_error_nonbool(var);
handle_ignore_submodules_arg(&default_diff_options, value); handle_ignore_submodules_arg(&default_diff_options, value);
}


if (!strcmp(var, "diff.submodule")) { if (!strcmp(var, "diff.submodule")) {
if (!value)
return config_error_nonbool(var);
if (parse_submodule_params(&default_diff_options, value)) if (parse_submodule_params(&default_diff_options, value))
warning(_("Unknown value for 'diff.submodule' config variable: '%s'"), warning(_("Unknown value for 'diff.submodule' config variable: '%s'"),
value); value);
@ -473,7 +481,10 @@ int git_diff_basic_config(const char *var, const char *value,
} }


if (!strcmp(var, "diff.wserrorhighlight")) { if (!strcmp(var, "diff.wserrorhighlight")) {
int val = parse_ws_error_highlight(value); int val;
if (!value)
return config_error_nonbool(var);
val = parse_ws_error_highlight(value);
if (val < 0) if (val < 0)
return -1; return -1;
ws_error_highlight_default = val; ws_error_highlight_default = val;
@ -490,6 +501,8 @@ int git_diff_basic_config(const char *var, const char *value,


if (!strcmp(var, "diff.dirstat")) { if (!strcmp(var, "diff.dirstat")) {
struct strbuf errmsg = STRBUF_INIT; struct strbuf errmsg = STRBUF_INIT;
if (!value)
return config_error_nonbool(var);
default_diff_options.dirstat_permille = diff_dirstat_permille_default; default_diff_options.dirstat_permille = diff_dirstat_permille_default;
if (parse_dirstat_params(&default_diff_options, value, &errmsg)) if (parse_dirstat_params(&default_diff_options, value, &errmsg))
warning(_("Found errors in 'diff.dirstat' config variable:\n%s"), warning(_("Found errors in 'diff.dirstat' config variable:\n%s"),

View File

@ -1253,6 +1253,8 @@ static int git_mailinfo_config(const char *var, const char *value,
return 0; return 0;
} }
if (!strcmp(var, "mailinfo.quotedcr")) { if (!strcmp(var, "mailinfo.quotedcr")) {
if (!value)
return config_error_nonbool(var);
if (mailinfo_parse_quoted_cr_action(value, &mi->quoted_cr) != 0) if (mailinfo_parse_quoted_cr_action(value, &mi->quoted_cr) != 0)
return error(_("bad action '%s' for '%s'"), value, var); return error(_("bad action '%s' for '%s'"), value, var);
return 0; return 0;

View File

@ -112,6 +112,8 @@ static int notes_rewrite_config(const char *k, const char *v,
} }
return 0; return 0;
} else if (!c->refs_from_env && !strcmp(k, "notes.rewriteref")) { } else if (!c->refs_from_env && !strcmp(k, "notes.rewriteref")) {
if (!v)
return config_error_nonbool(k);
/* note that a refs/ prefix is implied in the /* note that a refs/ prefix is implied in the
* underlying for_each_glob_ref */ * underlying for_each_glob_ref */
if (starts_with(v, "refs/notes/")) if (starts_with(v, "refs/notes/"))

View File

@ -507,6 +507,8 @@ static int git_trailer_default_config(const char *conf_key, const char *value,
warning(_("unknown value '%s' for key '%s'"), warning(_("unknown value '%s' for key '%s'"),
value, conf_key); value, conf_key);
} else if (!strcmp(trailer_item, "separators")) { } else if (!strcmp(trailer_item, "separators")) {
if (!value)
return config_error_nonbool(conf_key);
separators = xstrdup(value); separators = xstrdup(value);
} }
} }