Merge branch 'rs/config-unit-parsing' into maint

The code to parse scaled numbers out of configuration files has
been made more robust and also easier to follow.

* rs/config-unit-parsing:
  config: simplify parsing of unit factors
  config: don't multiply in parse_unit_factor()
  config: use unsigned_mult_overflows to check for overflows
This commit is contained in:
Junio C Hamano 2019-07-29 12:38:19 -07:00
Родитель e59150199c 39c575c969
Коммит 97cb523f00
1 изменённых файлов: 18 добавлений и 21 удалений

Просмотреть файл

@ -834,22 +834,16 @@ static int git_parse_source(config_fn_t fn, void *data,
return error_return; return error_return;
} }
static int parse_unit_factor(const char *end, uintmax_t *val) static uintmax_t get_unit_factor(const char *end)
{ {
if (!*end) if (!*end)
return 1; return 1;
else if (!strcasecmp(end, "k")) { else if (!strcasecmp(end, "k"))
*val *= 1024; return 1024;
return 1; else if (!strcasecmp(end, "m"))
} return 1024 * 1024;
else if (!strcasecmp(end, "m")) { else if (!strcasecmp(end, "g"))
*val *= 1024 * 1024; return 1024 * 1024 * 1024;
return 1;
}
else if (!strcasecmp(end, "g")) {
*val *= 1024 * 1024 * 1024;
return 1;
}
return 0; return 0;
} }
@ -859,19 +853,20 @@ static int git_parse_signed(const char *value, intmax_t *ret, intmax_t max)
char *end; char *end;
intmax_t val; intmax_t val;
uintmax_t uval; uintmax_t uval;
uintmax_t factor = 1; uintmax_t factor;
errno = 0; errno = 0;
val = strtoimax(value, &end, 0); val = strtoimax(value, &end, 0);
if (errno == ERANGE) if (errno == ERANGE)
return 0; return 0;
if (!parse_unit_factor(end, &factor)) { factor = get_unit_factor(end);
if (!factor) {
errno = EINVAL; errno = EINVAL;
return 0; return 0;
} }
uval = val < 0 ? -val : val; uval = val < 0 ? -val : val;
uval *= factor; if (unsigned_mult_overflows(factor, uval) ||
if (uval > max || (val < 0 ? -val : val) > uval) { factor * uval > max) {
errno = ERANGE; errno = ERANGE;
return 0; return 0;
} }
@ -888,21 +883,23 @@ static int git_parse_unsigned(const char *value, uintmax_t *ret, uintmax_t max)
if (value && *value) { if (value && *value) {
char *end; char *end;
uintmax_t val; uintmax_t val;
uintmax_t oldval; uintmax_t factor;
errno = 0; errno = 0;
val = strtoumax(value, &end, 0); val = strtoumax(value, &end, 0);
if (errno == ERANGE) if (errno == ERANGE)
return 0; return 0;
oldval = val; factor = get_unit_factor(end);
if (!parse_unit_factor(end, &val)) { if (!factor) {
errno = EINVAL; errno = EINVAL;
return 0; return 0;
} }
if (val > max || oldval > val) { if (unsigned_mult_overflows(factor, val) ||
factor * val > max) {
errno = ERANGE; errno = ERANGE;
return 0; return 0;
} }
val *= factor;
*ret = val; *ret = val;
return 1; return 1;
} }