From cef054d5b2d29422b2160783aea6d0581c701483 Mon Sep 17 00:00:00 2001 From: marcandre Date: Tue, 6 Nov 2012 17:13:19 +0000 Subject: [PATCH] * enum.c (enum_cycle): Support for Enumerable#cycle.size [Feature #6636] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@37510 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- enum.c | 20 +++++++++++++++++++- test/ruby/test_enumerator.rb | 5 +++++ 2 files changed, 24 insertions(+), 1 deletion(-) diff --git a/enum.c b/enum.c index 75b18454c9..e9c925611d 100644 --- a/enum.c +++ b/enum.c @@ -2235,6 +2235,24 @@ cycle_i(VALUE i, VALUE ary, int argc, VALUE *argv) return Qnil; } +static VALUE +enum_cycle_size(VALUE self, VALUE args) +{ + long mul; + VALUE n = Qnil; + VALUE size = enum_size(self, args); + + if (size == Qnil) return Qnil; + + if (args && (RARRAY_LEN(args) > 0)) { + n = RARRAY_PTR(args)[0]; + } + if (n == Qnil) return DBL2NUM(INFINITY); + mul = NUM2LONG(n); + if (mul <= 0) return INT2FIX(0); + return rb_funcall(size, '*', 1, LONG2FIX(mul)); +} + /* * call-seq: * enum.cycle(n=nil) { |obj| block } -> nil @@ -2265,7 +2283,7 @@ enum_cycle(int argc, VALUE *argv, VALUE obj) rb_scan_args(argc, argv, "01", &nv); - RETURN_ENUMERATOR(obj, argc, argv); + RETURN_SIZED_ENUMERATOR(obj, argc, argv, enum_cycle_size); if (NIL_P(nv)) { n = -1; } diff --git a/test/ruby/test_enumerator.rb b/test/ruby/test_enumerator.rb index a8564c1ce0..66d5e28a14 100644 --- a/test/ruby/test_enumerator.rb +++ b/test/ruby/test_enumerator.rb @@ -477,6 +477,11 @@ class TestEnumerator < Test::Unit::TestCase assert_equal 0, [:foo, :bar].cycle(-10).size assert_equal 0, [].cycle.size assert_equal 0, [].cycle(5).size + + assert_equal nil, @obj.cycle.size + assert_equal nil, @obj.cycle(5).size + assert_equal Float::INFINITY, @sized.cycle.size + assert_equal 126, @sized.cycle(3).size end def test_size_for_loops