mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
* vm_method.c (rb_class_clear_method_cache): should clear all
RCLASS_CALLABLE_M_TBLs of all sub-classes (T_ICLASS). RCLASS_CALLABLE_M_TBL() caches complemented method entries. It should be cleared when the modules are cleared. On previous version clears only for direct children. It is enough for normal modules because corresponding T_ICLASSes are direct children. However, refinements create complex data structure. So that we need to clear all children (and descendants). [ruby-core:71423] [Bug #11672] * vm_method.c (rb_clear_method_cache_by_class): rb_mKernel doesn't call rb_class_clear_method_cache, so that clear child T_ICLASSes. * test/ruby/test_refinement.rb: enable disabled test. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52648 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
1f2b9a46a1
commit
fa3affad9a
3 changed files with 41 additions and 12 deletions
21
ChangeLog
21
ChangeLog
|
@ -1,3 +1,24 @@
|
||||||
|
Wed Nov 18 22:50:43 2015 Koichi Sasada <ko1@atdot.net>
|
||||||
|
|
||||||
|
* vm_method.c (rb_class_clear_method_cache): should clear all
|
||||||
|
RCLASS_CALLABLE_M_TBLs of all sub-classes (T_ICLASS).
|
||||||
|
|
||||||
|
RCLASS_CALLABLE_M_TBL() caches complemented method entries.
|
||||||
|
It should be cleared when the modules are cleared.
|
||||||
|
On previous version clears only for direct children.
|
||||||
|
It is enough for normal modules because corresponding T_ICLASSes
|
||||||
|
are direct children.
|
||||||
|
|
||||||
|
However, refinements create complex data structure. So that
|
||||||
|
we need to clear all children (and descendants).
|
||||||
|
[ruby-core:71423] [Bug #11672]
|
||||||
|
|
||||||
|
* vm_method.c (rb_clear_method_cache_by_class): rb_mKernel
|
||||||
|
doesn't call rb_class_clear_method_cache, so that
|
||||||
|
clear child T_ICLASSes.
|
||||||
|
|
||||||
|
* test/ruby/test_refinement.rb: enable disabled test.
|
||||||
|
|
||||||
Wed Nov 18 21:09:08 2015 Koichi Sasada <ko1@atdot.net>
|
Wed Nov 18 21:09:08 2015 Koichi Sasada <ko1@atdot.net>
|
||||||
|
|
||||||
* vm_method.c (prepare_callable_method_entry): use
|
* vm_method.c (prepare_callable_method_entry): use
|
||||||
|
|
|
@ -1495,7 +1495,6 @@ class TestRefinement < Test::Unit::TestCase
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_reopen_refinement_module
|
def test_reopen_refinement_module
|
||||||
flag = false
|
|
||||||
assert_separately([], <<-"end;")
|
assert_separately([], <<-"end;")
|
||||||
$VERBOSE = nil
|
$VERBOSE = nil
|
||||||
class C
|
class C
|
||||||
|
@ -1522,11 +1521,6 @@ class TestRefinement < Test::Unit::TestCase
|
||||||
|
|
||||||
assert_equal(:bar, C.new.m, "[ruby-core:71423] [Bug #11672]")
|
assert_equal(:bar, C.new.m, "[ruby-core:71423] [Bug #11672]")
|
||||||
end;
|
end;
|
||||||
flag = true
|
|
||||||
rescue MiniTest::Assertion
|
|
||||||
skip 'expected to fail'
|
|
||||||
ensure
|
|
||||||
raise MiniTest::Assertion, 'this test is expected to fail' if flag
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
26
vm_method.c
26
vm_method.c
|
@ -62,6 +62,20 @@ static void
|
||||||
rb_class_clear_method_cache(VALUE klass, VALUE arg)
|
rb_class_clear_method_cache(VALUE klass, VALUE arg)
|
||||||
{
|
{
|
||||||
RCLASS_SERIAL(klass) = rb_next_class_serial();
|
RCLASS_SERIAL(klass) = rb_next_class_serial();
|
||||||
|
|
||||||
|
if (RB_TYPE_P(klass, T_ICLASS)) {
|
||||||
|
struct rb_id_table *table = RCLASS_CALLABLE_M_TBL(klass);
|
||||||
|
if (table) {
|
||||||
|
rb_id_table_clear(table);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (RCLASS_CALLABLE_M_TBL(klass) != 0) {
|
||||||
|
rb_obj_info_dump(klass);
|
||||||
|
rb_bug("RCLASS_CALLABLE_M_TBL(klass) != 0");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
rb_class_foreach_subclass(klass, rb_class_clear_method_cache, arg);
|
rb_class_foreach_subclass(klass, rb_class_clear_method_cache, arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,14 +107,14 @@ rb_clear_method_cache_by_class(VALUE klass)
|
||||||
else {
|
else {
|
||||||
rb_class_clear_method_cache(klass, Qnil);
|
rb_class_clear_method_cache(klass, Qnil);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (RB_TYPE_P(klass, T_MODULE)) {
|
if (klass == rb_mKernel) {
|
||||||
rb_subclass_entry_t *entry = RCLASS_EXT(klass)->subclasses;
|
rb_subclass_entry_t *entry = RCLASS_EXT(klass)->subclasses;
|
||||||
|
|
||||||
for (; entry != NULL; entry = entry->next) {
|
for (; entry != NULL; entry = entry->next) {
|
||||||
struct rb_id_table *table = RCLASS_CALLABLE_M_TBL(entry->klass);
|
struct rb_id_table *table = RCLASS_CALLABLE_M_TBL(entry->klass);
|
||||||
if (table)rb_id_table_clear(table);
|
if (table)rb_id_table_clear(table);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue