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

ruby-changes:13187

From: marcandre <ko1@a...>
Date: Wed, 16 Sep 2009 06:40:42 +0900 (JST)
Subject: [ruby-changes:13187] Ruby:r24943 (trunk): * thread.c (rb_exec_recursive_outer, rb_exec_recursive): Added method to short-circuit to the outermost level in case of recursion

marcandre	2009-09-16 06:30:50 +0900 (Wed, 16 Sep 2009)

  New Revision: 24943

  http://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=rev&revision=24943

  Log:
    * thread.c (rb_exec_recursive_outer, rb_exec_recursive): Added method to short-circuit to the outermost level in case of recursion
    
    * test/ruby/test_thread.rb (test_recursive_outer): Test for above
    
    * hash.c (rb_hash_hash): Return a sensible hash for in case of recursion [ruby-core:24648]
    
    * range.c (rb_range_hash): ditto
    
    * struct.c (rb_struct_hash): ditto
    
    * array.c (rb_array_hash): ditto
    
    * test/ruby/test_array.rb (test_hash2): test for above

  Modified files:
    trunk/ChangeLog
    trunk/array.c
    trunk/hash.c
    trunk/include/ruby/intern.h
    trunk/range.c
    trunk/struct.c
    trunk/test/ruby/test_array.rb
    trunk/test/ruby/test_thread.rb
    trunk/thread.c

Index: array.c
===================================================================
--- array.c	(revision 24942)
+++ array.c	(revision 24943)
@@ -2884,13 +2884,15 @@
     st_index_t h;
     VALUE n;
 
+    h = rb_hash_start(RARRAY_LEN(ary));
     if (recur) {
-        rb_raise(rb_eArgError, "recursive key for hash");
+	h = rb_hash_uint(h, NUM2LONG(rb_hash(rb_cArray)));
     }
-    h = rb_hash_start(RARRAY_LEN(ary));
-    for (i=0; i<RARRAY_LEN(ary); i++) {
-	n = rb_hash(RARRAY_PTR(ary)[i]);
-	h = rb_hash_uint(h, NUM2LONG(n));
+    else {
+	for (i=0; i<RARRAY_LEN(ary); i++) {
+	    n = rb_hash(RARRAY_PTR(ary)[i]);
+	    h = rb_hash_uint(h, NUM2LONG(n));
+	}
     }
     h = rb_hash_end(h);
     return LONG2FIX(h);
@@ -2907,7 +2909,7 @@
 static VALUE
 rb_ary_hash(VALUE ary)
 {
-    return rb_exec_recursive(recursive_hash, ary, 0);
+    return rb_exec_recursive_outer(recursive_hash, ary, 0);
 }
 
 /*
Index: include/ruby/intern.h
===================================================================
--- include/ruby/intern.h	(revision 24942)
+++ include/ruby/intern.h	(revision 24943)
@@ -343,6 +343,7 @@
 void rb_thread_atfork_before_exec(void);
 VALUE rb_exec_recursive(VALUE(*)(VALUE, VALUE, int),VALUE,VALUE);
 VALUE rb_exec_recursive_paired(VALUE(*)(VALUE, VALUE, int),VALUE,VALUE,VALUE);
+VALUE rb_exec_recursive_outer(VALUE(*)(VALUE, VALUE, int),VALUE,VALUE);
 /* file.c */
 VALUE rb_file_s_expand_path(int, VALUE *);
 VALUE rb_file_expand_path(VALUE, VALUE);
Index: ChangeLog
===================================================================
--- ChangeLog	(revision 24942)
+++ ChangeLog	(revision 24943)
@@ -1,3 +1,21 @@
+Wed Sep 16 06:30:07 2009  Marc-Andre Lafortune  <ruby-core@m...>
+
+	* thread.c (rb_exec_recursive_outer, rb_exec_recursive): Added method
+	  to short-circuit to the outermost level in case of recursion
+	
+	* test/ruby/test_thread.rb (test_recursive_outer): Test for above
+	
+	* hash.c (rb_hash_hash): Return a sensible hash for in case of
+	  recursion [ruby-core:24648]
+	
+	* range.c (rb_range_hash): ditto
+	
+	* struct.c (rb_struct_hash): ditto
+	
+	* array.c (rb_array_hash): ditto
+	
+	* test/ruby/test_array.rb (test_hash2): test for above
+
 Wed Sep 16 06:17:33 2009  Marc-Andre Lafortune  <ruby-core@m...>
 
 	* vm_eval.c (rb_catch_obj, rb_catch, rb_f_catch): No longer use the
Index: range.c
===================================================================
--- range.c	(revision 24942)
+++ range.c	(revision 24943)
@@ -207,14 +207,13 @@
     st_index_t hash = EXCL(range);
     VALUE v;
 
-    if (recur) {
-        rb_raise(rb_eArgError, "recursive key for hash");
-    }
     hash = rb_hash_start(hash);
-    v = rb_hash(RANGE_BEG(range));
-    hash = rb_hash_uint(hash, NUM2LONG(v));
-    v = rb_hash(RANGE_END(range));
-    hash = rb_hash_uint(hash, NUM2LONG(v));
+    if (!recur) {
+	v = rb_hash(RANGE_BEG(range));
+	hash = rb_hash_uint(hash, NUM2LONG(v));
+	v = rb_hash(RANGE_END(range));
+	hash = rb_hash_uint(hash, NUM2LONG(v));
+    }
     hash = rb_hash_uint(hash, EXCL(range) << 24);
     hash = rb_hash_end(hash);
 
@@ -233,7 +232,7 @@
 static VALUE
 range_hash(VALUE range)
 {
-    return rb_exec_recursive(recursive_hash, range, 0);
+    return rb_exec_recursive_outer(recursive_hash, range, 0);
 }
 
 static void
Index: thread.c
===================================================================
--- thread.c	(revision 24942)
+++ thread.c	(revision 24943)
@@ -3494,33 +3494,77 @@
     rb_hash_delete(list, obj);
 }
 
+struct exec_recursive_params {
+    VALUE (*func) (VALUE, VALUE, int);
+    VALUE list;
+    VALUE obj;
+    VALUE objid;
+    VALUE pairid;
+    VALUE arg;
+};
+
+static VALUE
+exec_recursive_i(VALUE tag, struct exec_recursive_params *p)
+{
+    VALUE result = Qundef;
+    int state;
+
+    recursive_push(p->list, p->objid, p->pairid);
+    PUSH_TAG();
+    if ((state = EXEC_TAG()) == 0) {
+	result = (*p->func) (p->obj, p->arg, Qfalse);
+    }
+    POP_TAG();
+    recursive_pop(p->list, p->objid, p->pairid);
+    if (state)
+	JUMP_TAG(state);
+    return result;
+}
+
 /*
  * Calls func(obj, arg, recursive), where recursive is non-zero if the
  * current method is called recursively on obj, or on the pair <obj, pairid>
+ * If outer is 0, then the innermost func will be called with recursive set
+ * to Qtrue, otherwise the outermost func will be called. In the latter case,
+ * all inner func are short-circuited by throw.
+ * Implementation details: the value thrown is the recursive list which is
+ * proper to the current method and unlikely to be catched anywhere else.
+ * list[recursive_key] is used as a flag for the outermost call.
  */
 
 static VALUE
-exec_recursive(VALUE (*func) (VALUE, VALUE, int), VALUE obj, VALUE pairid, VALUE arg)
+exec_recursive(VALUE (*func) (VALUE, VALUE, int), VALUE obj, VALUE pairid, VALUE arg, int outer)
 {
-    VALUE list = recursive_list_access();
-    VALUE objid = rb_obj_id(obj);
+    struct exec_recursive_params p;
+    int outermost;
+    p.list = recursive_list_access();
+    p.objid = rb_obj_id(obj);
+    outermost = outer && !recursive_check(p.list, ID2SYM(recursive_key), 0);
 
-    if (recursive_check(list, objid, pairid)) {
+    if (recursive_check(p.list, p.objid, pairid)) {
+	if (outer && !outermost) {
+	    rb_throw_obj(p.list, p.list);
+	}
 	return (*func) (obj, arg, Qtrue);
     }
     else {
 	VALUE result = Qundef;
-	int state;
+	p.func = func;
+	p.obj = obj;
+	p.pairid = pairid;
+	p.arg = arg;
 
-	recursive_push(list, objid, pairid);
-	PUSH_TAG();
-	if ((state = EXEC_TAG()) == 0) {
-	    result = (*func) (obj, arg, Qfalse);
+	if (outermost) {
+	    recursive_push(p.list, ID2SYM(recursive_key), 0);
+	    result = rb_catch_obj(p.list, exec_recursive_i, (VALUE)&p);
+	    recursive_pop(p.list, ID2SYM(recursive_key), 0);
+	    if (result == p.list) {
+		result = (*func) (obj, arg, Qtrue);
+	    }
 	}
-	POP_TAG();
-	recursive_pop(list, objid, pairid);
-	if (state)
-	    JUMP_TAG(state);
+	else {
+	    result = exec_recursive_i(0, &p);
+	}
 	return result;
     }
 }
@@ -3533,21 +3577,32 @@
 VALUE
 rb_exec_recursive(VALUE (*func) (VALUE, VALUE, int), VALUE obj, VALUE arg)
 {
-    return exec_recursive(func, obj, 0, arg);
+    return exec_recursive(func, obj, 0, arg, 0);
 }
 
 /*
  * Calls func(obj, arg, recursive), where recursive is non-zero if the
- * current method is called recursively on the pair <obj, paired_obj>
- * (in that order)
+ * current method is called recursively on the ordered pair <obj, paired_obj>
  */
 
 VALUE
 rb_exec_recursive_paired(VALUE (*func) (VALUE, VALUE, int), VALUE obj, VALUE paired_obj, VALUE arg)
 {
-    return exec_recursive(func, obj, rb_obj_id(paired_obj), arg);
+    return exec_recursive(func, obj, rb_obj_id(paired_obj), arg, 0);
 }
 
+/*
+ * If recursion is detected on the current method and obj, the outermost
+ * func will be called with (obj, arg, Qtrue). All inner func will be
+ * short-circuited using throw.
+ */
+
+VALUE
+rb_exec_recursive_outer(VALUE (*func) (VALUE, VALUE, int), VALUE obj, VALUE arg)
+{
+    return exec_recursive(func, obj, 0, arg, 1);
+}
+
 /* tracer */
 
 static rb_event_hook_t *
Index: struct.c
===================================================================
--- struct.c	(revision 24942)
+++ struct.c	(revision 24943)
@@ -810,13 +810,12 @@
     st_index_t h;
     VALUE n;
 
-    if (recur) {
-        rb_raise(rb_eArgError, "recursive key for hash");
-    }
     h = rb_hash_start(rb_hash(rb_obj_class(s)));
-    for (i = 0; i < RSTRUCT_LEN(s); i++) {
-	n = rb_hash(RSTRUCT_PTR(s)[i]);
-	h = rb_hash_uint(h, NUM2LONG(n));
+    if (!recur) {
+	for (i = 0; i < RSTRUCT_LEN(s); i++) {
+	    n = rb_hash(RSTRUCT_PTR(s)[i]);
+	    h = rb_hash_uint(h, NUM2LONG(n));
+	}
     }
     h = rb_hash_end(h);
     return INT2FIX(h);
@@ -832,7 +831,7 @@
 static VALUE
 rb_struct_hash(VALUE s)
 {
-    return rb_exec_recursive(recursive_hash, s, 0);
+    return rb_exec_recursive_outer(recursive_hash, s, 0);
 }
 
 /*
Index: hash.c
===================================================================
--- hash.c	(revision 24942)
+++ hash.c	(revision 24943)
@@ -1556,13 +1556,13 @@
 {
     st_index_t hval;
 
-    if (recur) {
-	rb_raise(rb_eArgError, "recursive key for hash");
-    }
     if (!RHASH(hash)->ntbl)
         return LONG2FIX(0);
     hval = RHASH(hash)->ntbl->num_entries;
-    rb_hash_foreach(hash, hash_i, (VALUE)&hval);
+    if (recur)
+	hval = rb_hash_end(rb_hash_uint(rb_hash_start(rb_hash(rb_cHash)), hval));
+    else
+	rb_hash_foreach(hash, hash_i, (VALUE)&hval);
     return INT2FIX(hval);
 }
 
@@ -1577,7 +1577,7 @@
 static VALUE
 rb_hash_hash(VALUE hash)
 {
-    return rb_exec_recursive(recursive_hash, hash, 0);
+    return rb_exec_recursive_outer(recursive_hash, hash, 0);
 }
 
 static int
Index: test/ruby/test_array.rb
===================================================================
--- test/ruby/test_array.rb	(revision 24942)
+++ test/ruby/test_array.rb	(revision 24943)
@@ -1572,7 +1572,8 @@
   def test_hash2
     a = []
     a << a
-    assert_raise(ArgumentError) { a.hash }
+    assert_equal([[a]].hash, a.hash)
+    assert_not_equal([a, a].hash, a.hash) # Implementation dependent
   end
 
   def test_flatten2
Index: test/ruby/test_thread.rb
===================================================================
--- test/ruby/test_thread.rb	(revision 24942)
+++ test/ruby/test_thread.rb	(revision 24943)
@@ -458,6 +458,19 @@
     end
     assert_raise(TypeError) { [o].inspect }
   end
+
+  def test_recursive_outer
+    arr = []
+    obj = Struct.new(:foo, :visited).new(arr, false)
+    arr << obj
+    def obj.hash
+      self[:visited] = true
+      super
+      raise "recursive_outer should short circuit intermediate calls"
+    end
+    assert_nothing_raised {arr.hash}
+    assert(obj[:visited])
+  end
 end
 
 class TestThreadGroup < Test::Unit::TestCase

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

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