diff options
author | Otto Moerbeek <otto@cvs.openbsd.org> | 2017-12-01 19:04:16 +0000 |
---|---|---|
committer | Otto Moerbeek <otto@cvs.openbsd.org> | 2017-12-01 19:04:16 +0000 |
commit | 2165721de04821eb27ebeb6c0eb2391f7fb2fc0e (patch) | |
tree | f0dc7577aaf9bfe58bba039b06cdfd5053576aea | |
parent | 899968f518971ddb53ac4c372df97ba94206649e (diff) |
fix buglet in split_number() and optimize count_digits();
from kshe with a twist from myself; ok tb@
-rw-r--r-- | usr.bin/dc/bcode.c | 68 |
1 files changed, 50 insertions, 18 deletions
diff --git a/usr.bin/dc/bcode.c b/usr.bin/dc/bcode.c index 307c4aa7e60..adc132841de 100644 --- a/usr.bin/dc/bcode.c +++ b/usr.bin/dc/bcode.c @@ -1,4 +1,4 @@ -/* $OpenBSD: bcode.c,v 1.56 2017/11/29 19:13:31 otto Exp $ */ +/* $OpenBSD: bcode.c,v 1.57 2017/12/01 19:04:15 otto Exp $ */ /* * Copyright (c) 2003, Otto Moerbeek <otto@drijf.net> @@ -354,7 +354,7 @@ scale_number(BIGNUM *n, int s) abs_scale = s > 0 ? s : -s; - if (abs_scale < sizeof(factors)/sizeof(factors[0])) { + if (abs_scale < nitems(factors)) { if (s > 0) bn_check(BN_mul_word(n, factors[abs_scale])); else @@ -390,9 +390,10 @@ split_number(const struct number *n, BIGNUM *i, BIGNUM *f) bn_checkp(BN_copy(i, n->number)); - if (n->scale == 0 && f != NULL) - bn_check(BN_set_word(f, 0)); - else if (n->scale < sizeof(factors)/sizeof(factors[0])) { + if (n->scale == 0) { + if (f != NULL) + bn_check(BN_set_word(f, 0)); + } else if (n->scale < nitems(factors)) { rem = BN_div_word(i, factors[n->scale]); if (f != NULL) bn_check(BN_set_word(f, rem)); @@ -697,25 +698,56 @@ push_scale(void) static u_int count_digits(const struct number *n) { - struct number *int_part, *fract_part; - u_int i; + BIGNUM *int_part, *a, *p; + BN_CTX *ctx; + uint d; + const uint64_t c = 1292913986; /* floor(2^32 * log_10(2)) */ + int bits; if (BN_is_zero(n->number)) return n->scale ? n->scale : 1; - int_part = new_number(); - fract_part = new_number(); - fract_part->scale = n->scale; - split_number(n, int_part->number, fract_part->number); + int_part = BN_new(); + bn_checkp(int_part); + + split_number(n, int_part, NULL); + bits = BN_num_bits(int_part); + + if (bits == 0) + d = 0; + else { + /* + * Estimate number of decimal digits based on number of bits. + * Divide 2^32 factor out by shifting + */ + d = (c * bits) >> 32; + + /* If close to a possible rounding error fix if needed */ + if (d != (c * (bits - 1)) >> 32) { + ctx = BN_CTX_new(); + bn_checkp(ctx); + a = BN_new(); + bn_checkp(a); + p = BN_new(); + bn_checkp(p); - i = 0; - while (!BN_is_zero(int_part->number)) { - (void)BN_div_word(int_part->number, 10); - i++; + bn_check(BN_set_word(a, 10)); + bn_check(BN_set_word(p, d)); + bn_check(BN_exp(a, a, p, ctx)); + + if (BN_ucmp(int_part, a) >= 0) + d++; + + BN_CTX_free(ctx); + BN_free(a); + BN_free(p); + } else + d++; } - free_number(int_part); - free_number(fract_part); - return i + n->scale; + + BN_free(int_part); + + return d + n->scale; } static void |