diff options
author | Kenta Murata <mrkn@mrkn.jp> | 2021-01-06 10:25:45 +0900 |
---|---|---|
committer | Kenta Murata <mrkn@mrkn.jp> | 2021-01-06 10:56:48 +0900 |
commit | f289f8ae3ab3e76287597722c52cd8cfcfc694ad (patch) | |
tree | 76a9fbdaf61b8261405adbc003397b7be7ff9a34 | |
parent | 31854403b3398e1c7fa642dca9d2dfe02b171371 (diff) |
[ruby/bigdecimal] Optimize the conversion from small Bignum
https://github.com/ruby/bigdecimal/commit/4792a917d8
-rw-r--r-- | ext/bigdecimal/bigdecimal.c | 28 |
1 files changed, 26 insertions, 2 deletions
diff --git a/ext/bigdecimal/bigdecimal.c b/ext/bigdecimal/bigdecimal.c index b4f71142e8..a912908664 100644 --- a/ext/bigdecimal/bigdecimal.c +++ b/ext/bigdecimal/bigdecimal.c @@ -2763,8 +2763,32 @@ rb_int64_convert_to_BigDecimal(int64_t ival, size_t digs, int raise_exception) static VALUE rb_big_convert_to_BigDecimal(VALUE val, RB_UNUSED_VAR(size_t digs), int raise_exception) { - Real *vp = GetVpValue(val, 1); - return check_exception(vp->obj); + assert(RB_TYPE_P(val, T_BIGNUM)); + + size_t size = rb_absint_size(val, NULL); + int sign = rb_big_cmp(val, INT2FIX(0)); + if (size <= sizeof(long)) { + if (sign < 0) { + return rb_int64_convert_to_BigDecimal(NUM2LONG(val), digs, raise_exception); + } + else { + return rb_uint64_convert_to_BigDecimal(NUM2ULONG(val), digs, raise_exception); + } + } +#if defined(SIZEOF_LONG_LONG) && SIZEOF_LONG < SIZEOF_LONG_LONG + else if (size <= sizeof(LONG_LONG)) { + if (sign < 0) { + return rb_int64_convert_to_BigDecimal(NUM2LL(val), digs, raise_exception); + } + else { + return rb_uint64_convert_to_BigDecimal(NUM2ULL(val), digs, raise_exception); + } + } +#endif + else { + Real *vp = GetVpValue(val, 1); + return check_exception(vp->obj); + } } static VALUE |