[前][次][番号順一覧][スレッド一覧]

ruby-changes:51440

From: mame <ko1@a...>
Date: Wed, 13 Jun 2018 18:04:38 +0900 (JST)
Subject: [ruby-changes:51440] mame:r63646 (trunk): range.c: prohibit `(1..nil)`

mame	2018-06-13 18:04:32 +0900 (Wed, 13 Jun 2018)

  New Revision: 63646

  https://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=revision&revision=63646

  Log:
    range.c: prohibit `(1..nil)`
    
    Now endless range can be created by either a literal `(1..)` or explicit
    range creation `Range.new(1, nil)`.  [Bug #14845]
    
    This change is intended for "early failure"; for example,
    `(1..var).to_a` causes out of memory if `var` is inadvertently nil.

  Modified files:
    trunk/compile.c
    trunk/parse.y
    trunk/range.c
    trunk/test/ruby/test_range.rb
Index: range.c
===================================================================
--- range.c	(revision 63645)
+++ range.c	(revision 63646)
@@ -37,7 +37,7 @@ static VALUE r_cover_p(VALUE, VALUE, VAL https://github.com/ruby/ruby/blob/trunk/range.c#L37
 static void
 range_init(VALUE range, VALUE beg, VALUE end, VALUE exclude_end)
 {
-    if ((!FIXNUM_P(beg) || !FIXNUM_P(end)) && !NIL_P(end)) {
+    if ((!FIXNUM_P(beg) || !FIXNUM_P(end)) && end != Qundef) {
 	VALUE v;
 
 	v = rb_funcall(beg, id_cmp, 1, end);
@@ -47,15 +47,16 @@ range_init(VALUE range, VALUE beg, VALUE https://github.com/ruby/ruby/blob/trunk/range.c#L47
 
     RANGE_SET_EXCL(range, exclude_end);
     RANGE_SET_BEG(range, beg);
-    RANGE_SET_END(range, end);
+    RANGE_SET_END(range, end == Qundef ? Qnil : end);
 }
 
 VALUE
-rb_range_new(VALUE beg, VALUE end, int exclude_end)
+rb_range_new(VALUE beg, VALUE end, int flag)
 {
     VALUE range = rb_obj_alloc(rb_cRange);
 
-    range_init(range, beg, end, RBOOL(exclude_end));
+    if (flag & 2) end = Qundef;
+    range_init(range, beg, end, RBOOL(flag & 1));
     return range;
 }
 
@@ -85,6 +86,7 @@ range_initialize(int argc, VALUE *argv, https://github.com/ruby/ruby/blob/trunk/range.c#L86
 
     rb_scan_args(argc, argv, "21", &beg, &end, &flags);
     range_modify(range);
+    if (NIL_P(end)) end = Qundef;
     range_init(range, beg, end, RBOOL(RTEST(flags)));
     return Qnil;
 }
@@ -1339,7 +1341,7 @@ range_dumper(VALUE range) https://github.com/ruby/ruby/blob/trunk/range.c#L1341
 
     rb_ivar_set(v, id_excl, RANGE_EXCL(range));
     rb_ivar_set(v, id_beg, RANGE_BEG(range));
-    rb_ivar_set(v, id_end, RANGE_END(range));
+    if (!NIL_P(RANGE_END(range))) rb_ivar_set(v, id_end, RANGE_END(range));
     return v;
 }
 
@@ -1354,7 +1356,7 @@ range_loader(VALUE range, VALUE obj) https://github.com/ruby/ruby/blob/trunk/range.c#L1356
 
     range_modify(range);
     beg = rb_ivar_get(obj, id_beg);
-    end = rb_ivar_get(obj, id_end);
+    end = rb_ivar_lookup(obj, id_end, Qundef);
     excl = rb_ivar_get(obj, id_excl);
     if (!NIL_P(excl)) {
 	range_init(range, beg, end, RBOOL(RTEST(excl)));
Index: parse.y
===================================================================
--- parse.y	(revision 63645)
+++ parse.y	(revision 63646)
@@ -1916,24 +1916,16 @@ arg		: lhs '=' arg_rhs https://github.com/ruby/ruby/blob/trunk/parse.y#L1916
 		| arg tDOT2
 		    {
 		    /*%%%*/
-                        YYLTYPE loc;
-                        loc.beg_pos = @2.end_pos;
-                        loc.end_pos = @2.end_pos;
-
 			value_expr($1);
-			$$ = NEW_DOT2($1, new_nil(&loc), &@$);
+			$$ = NEW_DOT2($1, 0, &@$);
 		    /*% %*/
 		    /*% ripper: dot2!($1, Qnil) %*/
 		    }
 		| arg tDOT3
 		    {
 		    /*%%%*/
-                        YYLTYPE loc;
-                        loc.beg_pos = @2.end_pos;
-                        loc.end_pos = @2.end_pos;
-
 			value_expr($1);
-			$$ = NEW_DOT3($1, new_nil(&loc), &@$);
+			$$ = NEW_DOT3($1, 0, &@$);
 		    /*% %*/
 		    /*% ripper: dot3!($1, Qnil) %*/
 		    }
Index: compile.c
===================================================================
--- compile.c	(revision 63645)
+++ compile.c	(revision 63646)
@@ -7188,10 +7188,10 @@ iseq_compile_each0(rb_iseq_t *iseq, LINK https://github.com/ruby/ruby/blob/trunk/compile.c#L7188
       case NODE_DOT2:
       case NODE_DOT3:{
 	int excl = type == NODE_DOT3;
-	VALUE flag = INT2FIX(excl);
 	const NODE *b = node->nd_beg;
 	const NODE *e = node->nd_end;
-	if (number_literal_p(b) && number_literal_p(e)) {
+	VALUE flag = INT2FIX(excl | (e ? 0 : 2));
+	if (number_literal_p(b) && e && number_literal_p(e)) {
 	    if (!popped) {
 		VALUE val = rb_range_new(b->nd_lit, e->nd_lit, excl);
 		iseq_add_mark_object_compile_time(iseq, val);
Index: test/ruby/test_range.rb
===================================================================
--- test/ruby/test_range.rb	(revision 63645)
+++ test/ruby/test_range.rb	(revision 63646)
@@ -13,8 +13,8 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L13
 
     assert_raise(ArgumentError) { (1.."3") }
 
-    assert_equal((0..nil), Range.new(0, nil, false))
-    assert_equal((0...nil), Range.new(0, nil, true))
+    assert_equal((0..), Range.new(0, nil, false))
+    assert_equal((0...), Range.new(0, nil, true))
 
     obj = Object.new
     def obj.<=>(other)
@@ -161,15 +161,15 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L161
     assert_not_equal(r, (1..2))
     assert_not_equal(r, (0..2))
     assert_not_equal(r, (0...1))
-    assert_not_equal(r, (0..nil))
+    assert_not_equal(r, (0..))
     subclass = Class.new(Range)
     assert_equal(r, subclass.new(0,1))
 
-    r = (0..nil)
+    r = (0..)
     assert_equal(r, r)
-    assert_equal(r, (0..nil))
+    assert_equal(r, (0..))
     assert_not_equal(r, 0)
-    assert_not_equal(r, (0...nil))
+    assert_not_equal(r, (0...))
     subclass = Class.new(Range)
     assert_equal(r, subclass.new(0,nil))
   end
@@ -185,11 +185,11 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L185
     subclass = Class.new(Range)
     assert_operator(r, :eql?, subclass.new(0,1))
 
-    r = (0..nil)
+    r = (0..)
     assert_operator(r, :eql?, r)
-    assert_operator(r, :eql?, 0..nil)
+    assert_operator(r, :eql?, 0..)
     assert_not_operator(r, :eql?, 0)
-    assert_not_operator(r, :eql?, 0...nil)
+    assert_not_operator(r, :eql?, 0...)
     subclass = Class.new(Range)
     assert_operator(r, :eql?, subclass.new(0,nil))
   end
@@ -198,8 +198,8 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L198
     assert_kind_of(Integer, (0..1).hash)
     assert_equal((0..1).hash, (0..1).hash)
     assert_not_equal((0..1).hash, (0...1).hash)
-    assert_equal((0..nil).hash, (0..nil).hash)
-    assert_not_equal((0..nil).hash, (0...nil).hash)
+    assert_equal((0..).hash, (0..).hash)
+    assert_not_equal((0..).hash, (0...).hash)
   end
 
   def test_step
@@ -380,9 +380,9 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L380
     assert_equal(0, (0..1).begin)
     assert_equal(1, (0..1).end)
     assert_equal(1, (0...1).end)
-    assert_equal(0, (0..nil).begin)
-    assert_equal(nil, (0..nil).end)
-    assert_equal(nil, (0...nil).end)
+    assert_equal(0, (0..).begin)
+    assert_equal(nil, (0..).end)
+    assert_equal(nil, (0...).end)
   end
 
   def test_first_last
@@ -402,17 +402,17 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L402
     assert_equal("c", ("a"..."c").last)
     assert_equal(0, (2...0).last)
 
-    assert_equal([0, 1, 2], (0..nil).first(3))
-    assert_equal(0, (0..nil).first)
-    assert_equal("a", ("a"..nil).first)
+    assert_equal([0, 1, 2], (0..).first(3))
+    assert_equal(0, (0..).first)
+    assert_equal("a", ("a"..).first)
     # XXX: How should (0...).last(3) behave?
   end
 
   def test_to_s
     assert_equal("0..1", (0..1).to_s)
     assert_equal("0...1", (0...1).to_s)
-    assert_equal("0..", (0..nil).to_s)
-    assert_equal("0...", (0...nil).to_s)
+    assert_equal("0..", (0..).to_s)
+    assert_equal("0...", (0...).to_s)
 
     bug11767 = '[ruby-core:71811] [Bug #11767]'
     assert_predicate(("0".taint.."1").to_s, :tainted?, bug11767)
@@ -423,8 +423,8 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L423
   def test_inspect
     assert_equal("0..1", (0..1).inspect)
     assert_equal("0...1", (0...1).inspect)
-    assert_equal("0..", (0..nil).inspect)
-    assert_equal("0...", (0...nil).inspect)
+    assert_equal("0..", (0..).inspect)
+    assert_equal("0...", (0...).inspect)
 
     bug11767 = '[ruby-core:71811] [Bug #11767]'
     assert_predicate(("0".taint.."1").inspect, :tainted?, bug11767)
@@ -435,8 +435,8 @@ class TestRange < Test::Unit::TestCase https://github.com/ruby/ruby/blob/trunk/test/ruby/test_range.rb#L435
   def test_eqq
     assert_operator(0..10, :===, 5)
     assert_not_operator(0..10, :===, 11)
-    assert_operator(5..nil, :===, 11)
-    assert_not_operator(5..nil, :===, 0)
+    assert_operator(5.., :===, 11)
+    assert_not_operator(5.., :===, 0)
   end
 
   def test_eqq_time

--
ML: ruby-changes@q...
Info: http://www.atdot.net/~ko1/quickml/

[前][次][番号順一覧][スレッド一覧]