summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ChangeLog5
-rw-r--r--test/ruby/test_refinement.rb24
-rw-r--r--vm_method.c5
3 files changed, 32 insertions, 2 deletions
diff --git a/ChangeLog b/ChangeLog
index d67a11a359..12c288519e 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+Tue Aug 7 11:35:37 2012 Shugo Maeda <shugo@ruby-lang.org>
+
+ * vm_method.c (rb_redefine_opt_method): use RCLASS_ORIGIN to avoid
+ SEGV when a module-prepended class is refined.
+
Tue Aug 7 10:46:37 2012 NAKAMURA Usaku <usa@ruby-lang.org>
* test/ruby/test_file_exhaustive.rb
diff --git a/test/ruby/test_refinement.rb b/test/ruby/test_refinement.rb
index 65789899c5..cc460894c4 100644
--- a/test/ruby/test_refinement.rb
+++ b/test/ruby/test_refinement.rb
@@ -300,5 +300,29 @@ class TestRefinement < Test::Unit::TestCase
assert_equal("m2#bar", m.call_bar("abc"))
assert_equal("m3#baz", m.call_baz("abc"))
end
+
+ def test_refine_prepended_class
+ m1 = Module.new {
+ def foo
+ super << :m1
+ end
+ }
+ c = Class.new {
+ prepend m1
+
+ def foo
+ [:c]
+ end
+ }
+ m2 = Module.new {
+ refine c do
+ def foo
+ super << :m2
+ end
+ end
+ }
+ obj = c.new
+ assert_equal([:c, :m1, :m2], m2.module_eval { obj.foo })
+ end
end
diff --git a/vm_method.c b/vm_method.c
index 13174dd38d..6582bd1716 100644
--- a/vm_method.c
+++ b/vm_method.c
@@ -1415,13 +1415,14 @@ rb_redefine_opt_method(VALUE klass, ID mid)
{
st_data_t data;
rb_method_entry_t *me = 0;
+ VALUE origin = RCLASS_ORIGIN(klass);
- if (!st_lookup(RCLASS_M_TBL(klass), mid, &data) ||
+ if (!st_lookup(RCLASS_M_TBL(origin), mid, &data) ||
!(me = (rb_method_entry_t *)data) ||
(!me->def || me->def->type == VM_METHOD_TYPE_UNDEF)) {
return;
}
- rb_vm_check_redefinition_opt_method(me, klass);
+ rb_vm_check_redefinition_opt_method(me, origin);
}
void