1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

refine parse_rat

* rational.c (read_num): return the exponent instead of the
  divisor, to get rid of huge bignums.

* rational.c (parse_rat): subtract exponents instead of reduction
  of powers.

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@65618 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
nobu 2018-11-08 02:25:44 +00:00
parent dcfb7f6d54
commit 17e9667fe9
2 changed files with 43 additions and 25 deletions

View file

@ -2336,13 +2336,13 @@ negate_num(VALUE num)
}
static int
read_num(const char **s, const char *const end, VALUE *num, VALUE *div)
read_num(const char **s, const char *const end, VALUE *num, VALUE *nexp)
{
VALUE fp = ONE, exp, fn = ZERO, n = ZERO;
int expsign = 0, ok = 0;
char *e;
*div = ONE;
*nexp = ZERO;
*num = ZERO;
if (*s < end && **s != '.') {
n = rb_int_parse_cstr(*s, end-*s, &e, NULL,
@ -2364,10 +2364,9 @@ read_num(const char **s, const char *const end, VALUE *num, VALUE *div)
return 1;
*s = e;
{
VALUE l = f_expt10(SIZET2NUM(count));
VALUE l = f_expt10(*nexp = SIZET2NUM(count));
n = n == ZERO ? fp : rb_int_plus(rb_int_mul(*num, l), fp);
*num = n;
*div = l;
fn = SIZET2NUM(count);
}
ok = 1;
@ -2384,18 +2383,12 @@ read_num(const char **s, const char *const end, VALUE *num, VALUE *div)
if (exp != ZERO) {
if (expsign == '-') {
if (fn != ZERO) exp = rb_int_plus(exp, fn);
*div = f_expt10(exp);
}
else {
if (fn != ZERO) exp = rb_int_minus(exp, fn);
if (INT_NEGATIVE_P(exp)) {
*div = f_expt10(negate_num(exp));
}
else {
if (exp != ZERO) *num = rb_int_mul(n, f_expt10(exp));
*div = ONE;
}
exp = negate_num(exp);
}
*nexp = exp;
}
}
@ -2414,22 +2407,21 @@ static VALUE
parse_rat(const char *s, const char *const e, int strict, int raise)
{
int sign;
VALUE num, den, ndiv, ddiv;
VALUE num, den, nexp, dexp;
s = skip_ws(s, e);
sign = read_sign(&s, e);
if (!read_num(&s, e, &num, &ndiv)) {
if (!read_num(&s, e, &num, &nexp)) {
if (strict) return Qnil;
return canonicalization ? ZERO : nurat_s_alloc(rb_cRational);
}
nurat_reduce(&num, &ndiv);
den = ndiv;
den = ONE;
if (s < e && *s == '/') {
s++;
if (!read_num(&s, e, &den, &ddiv)) {
if (!read_num(&s, e, &den, &dexp)) {
if (strict) return Qnil;
den = ndiv;
den = ONE;
}
else if (den == ZERO) {
if (!raise) return Qnil;
@ -2439,17 +2431,38 @@ parse_rat(const char *s, const char *const e, int strict, int raise)
return Qnil;
}
else {
nurat_reduce(&den, &ddiv);
nexp = rb_int_minus(nexp, dexp);
nurat_reduce(&num, &den);
nurat_reduce(&ndiv, &ddiv);
if (ndiv != ONE) den = rb_int_mul(den, ndiv);
if (ddiv != ONE) num = rb_int_mul(num, ddiv);
}
}
else if (strict && skip_ws(s, e) != e) {
return Qnil;
}
if (nexp != ZERO) {
if (INT_NEGATIVE_P(nexp)) {
VALUE mul;
if (!FIXNUM_P(nexp)) {
overflow:
return sign == '-' ? DBL2NUM(-HUGE_VAL) : DBL2NUM(HUGE_VAL);
}
mul = f_expt10(LONG2NUM(-FIX2LONG(nexp)));
if (RB_FLOAT_TYPE_P(mul)) goto overflow;
num = rb_int_mul(num, mul);
}
else {
VALUE div;
if (!FIXNUM_P(nexp)) {
underflow:
return sign == '-' ? DBL2NUM(-0.0) : DBL2NUM(+0.0);
}
div = f_expt10(nexp);
if (RB_FLOAT_TYPE_P(div)) goto underflow;
den = rb_int_mul(den, div);
}
nurat_reduce(&num, &den);
}
if (sign == '-') {
num = negate_num(num);
}
@ -2459,6 +2472,8 @@ parse_rat(const char *s, const char *const e, int strict, int raise)
return num;
}
#define FLOAT_ZERO_P(x) (rb_float_value(x) == 0.0)
static VALUE
string_to_r_strict(VALUE self, int raise)
{
@ -2473,7 +2488,7 @@ string_to_r_strict(VALUE self, int raise)
self);
}
if (RB_FLOAT_TYPE_P(num)) {
if (RB_FLOAT_TYPE_P(num) && !FLOAT_ZERO_P(num)) {
if (!raise) return Qnil;
rb_raise(rb_eFloatDomainError, "Infinity");
}
@ -2517,7 +2532,7 @@ string_to_r(VALUE self)
num = parse_rat(RSTRING_PTR(self), RSTRING_END(self), 0, TRUE);
if (RB_FLOAT_TYPE_P(num))
if (RB_FLOAT_TYPE_P(num) && !FLOAT_ZERO_P(num))
rb_raise(rb_eFloatDomainError, "Infinity");
return num;
}
@ -2529,7 +2544,7 @@ rb_cstr_to_rat(const char *s, int strict) /* for complex's internal */
num = parse_rat(s, s + strlen(s), strict, TRUE);
if (RB_FLOAT_TYPE_P(num))
if (RB_FLOAT_TYPE_P(num) && !FLOAT_ZERO_P(num))
rb_raise(rb_eFloatDomainError, "Infinity");
return num;
}

View file

@ -120,6 +120,9 @@ class Rational_Test < Test::Unit::TestCase
assert_raise_with_message(ArgumentError, /\u{221a 2668}/) {
Rational("\u{221a 2668}")
}
assert_warning('') {
assert_predicate(Rational('1e-99999999999999999999'), :zero?)
}
assert_raise(TypeError){Rational(Object.new)}
assert_raise(TypeError){Rational(Object.new, Object.new)}