proc.c: proc without env
* proc.c (rb_sym_to_proc): move from string.c and create a Proc with no environments. [ruby-core:71088] [Bug #11594] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52129 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
d0e8610582
commit
c2e8fb0f34
@ -1,3 +1,8 @@
|
|||||||
|
Thu Oct 15 13:37:23 2015 Nobuyoshi Nakada <nobu@ruby-lang.org>
|
||||||
|
|
||||||
|
* proc.c (rb_sym_to_proc): move from string.c and create a Proc
|
||||||
|
with no environments. [ruby-core:71088] [Bug #11594]
|
||||||
|
|
||||||
Thu Oct 15 01:57:03 2015 CHIKANAGA Tomoyuki <nagachika@ruby-lang.org>
|
Thu Oct 15 01:57:03 2015 CHIKANAGA Tomoyuki <nagachika@ruby-lang.org>
|
||||||
|
|
||||||
* test/objspace/test_objspace.rb
|
* test/objspace/test_objspace.rb
|
||||||
|
36
proc.c
36
proc.c
@ -1035,6 +1035,42 @@ rb_hash_proc(st_index_t hash, VALUE prc)
|
|||||||
return rb_hash_uint(hash, (st_index_t)proc->block.ep >> 16);
|
return rb_hash_uint(hash, (st_index_t)proc->block.ep >> 16);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VALUE
|
||||||
|
rb_sym_to_proc(VALUE sym)
|
||||||
|
{
|
||||||
|
static VALUE sym_proc_cache = Qfalse;
|
||||||
|
enum {SYM_PROC_CACHE_SIZE = 67};
|
||||||
|
VALUE proc;
|
||||||
|
long index;
|
||||||
|
ID id;
|
||||||
|
VALUE *aryp;
|
||||||
|
|
||||||
|
if (!sym_proc_cache) {
|
||||||
|
sym_proc_cache = rb_ary_tmp_new(SYM_PROC_CACHE_SIZE * 2);
|
||||||
|
rb_gc_register_mark_object(sym_proc_cache);
|
||||||
|
rb_ary_store(sym_proc_cache, SYM_PROC_CACHE_SIZE*2 - 1, Qnil);
|
||||||
|
}
|
||||||
|
|
||||||
|
id = SYM2ID(sym);
|
||||||
|
index = (id % SYM_PROC_CACHE_SIZE) << 1;
|
||||||
|
|
||||||
|
aryp = RARRAY_PTR(sym_proc_cache);
|
||||||
|
if (aryp[index] == sym) {
|
||||||
|
return aryp[index + 1];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
rb_proc_t *ptr;
|
||||||
|
VALUE ifunc = (VALUE)IFUNC_NEW(rb_sym_proc_call, (VALUE)id, 0);
|
||||||
|
proc = rb_proc_alloc(rb_cProc);
|
||||||
|
ptr = RTYPEDDATA_DATA(proc);
|
||||||
|
ptr->block.iseq = (rb_iseq_t *)ifunc;
|
||||||
|
ptr->block.proc = ifunc;
|
||||||
|
aryp[index] = sym;
|
||||||
|
aryp[index + 1] = proc;
|
||||||
|
return proc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* prc.hash -> integer
|
* prc.hash -> integer
|
||||||
|
29
string.c
29
string.c
@ -8960,6 +8960,7 @@ rb_sym_proc_call(VALUE args, VALUE sym, int argc, const VALUE *argv, VALUE passe
|
|||||||
return rb_funcall_with_block(obj, (ID)sym, argc - 1, argv + 1, passed_proc);
|
return rb_funcall_with_block(obj, (ID)sym, argc - 1, argv + 1, passed_proc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* sym.to_proc
|
* sym.to_proc
|
||||||
@ -8972,34 +8973,8 @@ rb_sym_proc_call(VALUE args, VALUE sym, int argc, const VALUE *argv, VALUE passe
|
|||||||
VALUE
|
VALUE
|
||||||
rb_sym_to_proc(VALUE sym)
|
rb_sym_to_proc(VALUE sym)
|
||||||
{
|
{
|
||||||
static VALUE sym_proc_cache = Qfalse;
|
|
||||||
enum {SYM_PROC_CACHE_SIZE = 67};
|
|
||||||
VALUE proc;
|
|
||||||
long index;
|
|
||||||
ID id;
|
|
||||||
VALUE *aryp;
|
|
||||||
|
|
||||||
if (!sym_proc_cache) {
|
|
||||||
sym_proc_cache = rb_ary_tmp_new(SYM_PROC_CACHE_SIZE * 2);
|
|
||||||
rb_gc_register_mark_object(sym_proc_cache);
|
|
||||||
rb_ary_store(sym_proc_cache, SYM_PROC_CACHE_SIZE*2 - 1, Qnil);
|
|
||||||
}
|
|
||||||
|
|
||||||
id = SYM2ID(sym);
|
|
||||||
index = (id % SYM_PROC_CACHE_SIZE) << 1;
|
|
||||||
|
|
||||||
aryp = RARRAY_PTR(sym_proc_cache);
|
|
||||||
if (aryp[index] == sym) {
|
|
||||||
return aryp[index + 1];
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
proc = rb_proc_new(rb_sym_proc_call, (VALUE)id);
|
|
||||||
rb_block_clear_env_self(proc);
|
|
||||||
aryp[index] = sym;
|
|
||||||
aryp[index + 1] = proc;
|
|
||||||
return proc;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
|
@ -147,6 +147,14 @@ class TestSymbol < Test::Unit::TestCase
|
|||||||
end;
|
end;
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_to_proc_call_with_symbol_proc
|
||||||
|
first = 1
|
||||||
|
bug11594 = "[ruby-core:71088] [Bug #11594] corrupted the first local variable"
|
||||||
|
# symbol which does not have a Proc
|
||||||
|
->(&blk) {}.call(&:test_to_proc_call_with_symbol_proc)
|
||||||
|
assert_equal(1, first, bug11594)
|
||||||
|
end
|
||||||
|
|
||||||
def test_call
|
def test_call
|
||||||
o = Object.new
|
o = Object.new
|
||||||
def o.foo(x, y); x + y; end
|
def o.foo(x, y); x + y; end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user