aboutsummaryrefslogtreecommitdiffstats
path: root/rational.c
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-02-24 02:08:36 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-02-24 02:08:36 +0000
commit151fdd50fe1dcf7aadd170bad03a93e59015bc73 (patch)
treee798916262b7a73f6b716df8d4ed7aa9929d2e65 /rational.c
parent201380b5c4b22b08604acfbce2e51efdf4f87bc6 (diff)
downloadruby-151fdd50fe1dcf7aadd170bad03a93e59015bc73.tar.gz
rational.c: segfault on Rational exponent
* rational.c (read_num): fix segfault on Rational() with positive but less than the length of fractional part exponent. should be negated to convert to divisor which is a reciprocal. [ruby-core:85783] [Bug #14547] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@62555 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'rational.c')
-rw-r--r--rational.c22
1 files changed, 14 insertions, 8 deletions
diff --git a/rational.c b/rational.c
index 3fbfc7f1b1..b57677031d 100644
--- a/rational.c
+++ b/rational.c
@@ -2368,6 +2368,18 @@ islettere(int c)
return (c == 'e' || c == 'E');
}
+static VALUE
+negate_num(VALUE num)
+{
+ if (FIXNUM_P(num)) {
+ return rb_int_uminus(num);
+ }
+ else {
+ BIGNUM_NEGATE(num);
+ return rb_big_norm(num);
+ }
+}
+
static int
read_num(const char **s, const char *const end, VALUE *num, VALUE *div)
{
@@ -2422,7 +2434,7 @@ read_num(const char **s, const char *const end, VALUE *num, VALUE *div)
else {
if (fn != ZERO) exp = rb_int_minus(exp, fn);
if (INT_NEGATIVE_P(exp)) {
- *div = f_expt10(exp);
+ *div = f_expt10(negate_num(exp));
}
else {
*num = rb_int_mul(n, f_expt10(exp));
@@ -2483,13 +2495,7 @@ parse_rat(const char *s, const char *const e, int strict)
}
if (sign == '-') {
- if (FIXNUM_P(num)) {
- num = rb_int_uminus(num);
- }
- else {
- BIGNUM_NEGATE(num);
- num = rb_big_norm(num);
- }
+ num = negate_num(num);
}
if (!canonicalization || den != ONE)