summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authornagachika <nagachika@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2014-03-02 16:10:47 +0000
committernagachika <nagachika@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2014-03-02 16:10:47 +0000
commit24df9a76903702303b2fb6122bbd8e7322a4e7fa (patch)
tree74a34457660eab41bc215e0299a8a3f0f25de92d
parent648eb6784b7935e9edb260cb1577bdaf8be771e6 (diff)
merge revision(s) r45187,r45205,r45212,r45213: [Backport #9570]
* numeric.c (ruby_num_interval_step_size): check signs and get rid of implementation dependent behavior of negative division. [ruby-core:61106] [Bug #9570] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_2_0_0@45250 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog6
-rw-r--r--numeric.c18
-rw-r--r--test/ruby/test_numeric.rb19
-rw-r--r--version.h2
4 files changed, 37 insertions, 8 deletions
diff --git a/ChangeLog b/ChangeLog
index c719dfb0d9..f78a5f4573 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,9 @@
+Mon Mar 3 01:00:00 2014 Nobuyoshi Nakada <nobu@ruby-lang.org>
+
+ * numeric.c (ruby_num_interval_step_size): check signs and get rid
+ of implementation dependent behavior of negative division.
+ [ruby-core:61106] [Bug #9570]
+
Mon Mar 3 00:43:33 2014 CHIKANAGA Tomoyuki <nagachika@ruby-lang.org>
(merged partially from r42781)
diff --git a/numeric.c b/numeric.c
index d37edd719f..2c15e2fbe9 100644
--- a/numeric.c
+++ b/numeric.c
@@ -1816,21 +1816,29 @@ VALUE
num_interval_step_size(VALUE from, VALUE to, VALUE step, int excl)
{
if (FIXNUM_P(from) && FIXNUM_P(to) && FIXNUM_P(step)) {
- long delta, diff, result;
+ long delta, diff;
diff = FIX2LONG(step);
+ if (!diff) rb_num_zerodiv();
delta = FIX2LONG(to) - FIX2LONG(from);
+ if (diff < 0) {
+ diff = -diff;
+ delta = -delta;
+ }
if (excl) {
- delta += (diff > 0 ? -1 : +1);
+ delta--;
+ }
+ if (delta < 0) {
+ return INT2FIX(0);
}
- result = delta / diff;
- return LONG2FIX(result >= 0 ? result + 1 : 0);
+ return ULONG2NUM(delta / diff + 1UL);
}
else if (RB_TYPE_P(from, T_FLOAT) || RB_TYPE_P(to, T_FLOAT) || RB_TYPE_P(step, T_FLOAT)) {
double n = ruby_float_step_size(NUM2DBL(from), NUM2DBL(to), NUM2DBL(step), excl);
if (isinf(n)) return DBL2NUM(n);
- return LONG2FIX(n);
+ if (POSFIXABLE(n)) return LONG2FIX(n);
+ return rb_dbl2big(n);
}
else {
VALUE result;
diff --git a/test/ruby/test_numeric.rb b/test/ruby/test_numeric.rb
index b99ce988a1..66f11ab801 100644
--- a/test/ruby/test_numeric.rb
+++ b/test/ruby/test_numeric.rb
@@ -243,17 +243,32 @@ class TestNumeric < Test::Unit::TestCase
end
def test_step
+ i, bignum = 32, 1 << 30
+ bignum <<= (i <<= 1) - 32 until bignum.is_a?(Bignum)
assert_raise(ArgumentError) { 1.step(10, 1, 0) { } }
assert_raise(ArgumentError) { 1.step(10, 0) { } }
+ assert_equal(bignum*2+1, (-bignum).step(bignum, 1).size)
+ assert_equal(bignum*2, (-bignum).step(bignum-1, 1).size)
+
+ assert_equal(10+1, (0.0).step(10.0, 1.0).size)
+
+ i, bigflo = 1, bignum.to_f
+ i <<= 1 until (bigflo - i).to_i < bignum
+ bigflo -= i >> 1
+ assert_equal(bigflo.to_i, (0.0).step(bigflo-1.0, 1.0).size)
+ assert_operator((0.0).step(bignum.to_f, 1.0).size, :>=, bignum) # may loose precision
+
assert_step [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], [1, 10]
assert_step [1, 3, 5, 7, 9], [1, 10, 2]
assert_step [10, 8, 6, 4, 2], [10, 1, -2]
assert_step [1.0, 3.0, 5.0, 7.0, 9.0], [1.0, 10.0, 2.0]
- assert_step [1], [1, 10, 2**32]
+ assert_step [1], [1, 10, bignum]
- assert_step [10], [10, 1, -(2**32)]
+ assert_step [], [2, 1, 3]
+ assert_step [], [-2, -1, -3]
+ assert_step [10], [10, 1, -(bignum)]
assert_step [], [1, 0, Float::INFINITY]
assert_step [], [0, 1, -Float::INFINITY]
diff --git a/version.h b/version.h
index 8e446ea6b3..dd52021e6a 100644
--- a/version.h
+++ b/version.h
@@ -1,6 +1,6 @@
#define RUBY_VERSION "2.0.0"
#define RUBY_RELEASE_DATE "2014-03-03"
-#define RUBY_PATCHLEVEL 453
+#define RUBY_PATCHLEVEL 454
#define RUBY_RELEASE_YEAR 2014
#define RUBY_RELEASE_MONTH 3