aboutsummaryrefslogtreecommitdiffstats
path: root/class.c
diff options
context:
space:
mode:
authorcharliesome <charliesome@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2012-12-17 09:49:00 +0000
committercharliesome <charliesome@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2012-12-17 09:49:00 +0000
commita70bb8889b3130d72ec5a1f0ce92f386000a60da (patch)
tree4753f70eb20da004c9f294143593787f280c8cc6 /class.c
parent6d8ad6c489f41dfb5553d0933793eb5c84e0cc29 (diff)
downloadruby-a70bb8889b3130d72ec5a1f0ce92f386000a60da.tar.gz
* class.c (rewrite_cref_stack, clone_method): rewrite a method's cref
stack when cloning into a new class to allow lexical const lookup to work as expected [ruby-core:47834] [Bug #7107] * test/ruby/test_class.rb (class TestClass): related test git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@38423 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'class.c')
-rw-r--r--class.c18
1 files changed, 18 insertions, 0 deletions
diff --git a/class.c b/class.c
index 43862d6d4d..1fc7172d27 100644
--- a/class.c
+++ b/class.c
@@ -121,6 +121,23 @@ rb_class_new(VALUE super)
return rb_class_boot(super);
}
+static NODE*
+rewrite_cref_stack(NODE *node, VALUE old_klass, VALUE new_klass)
+{
+ NODE *new_node;
+ if (!node) {
+ return NULL;
+ }
+ if (node->nd_clss == old_klass) {
+ new_node = NEW_CREF(new_klass);
+ new_node->nd_next = node->nd_next;
+ } else {
+ new_node = NEW_CREF(node->nd_clss);
+ new_node->nd_next = rewrite_cref_stack(node->nd_next, old_klass, new_klass);
+ }
+ return new_node;
+}
+
static void
clone_method(VALUE klass, ID mid, const rb_method_entry_t *me)
{
@@ -129,6 +146,7 @@ clone_method(VALUE klass, ID mid, const rb_method_entry_t *me)
rb_iseq_t *iseq;
newiseqval = rb_iseq_clone(me->def->body.iseq->self, klass);
GetISeqPtr(newiseqval, iseq);
+ iseq->cref_stack = rewrite_cref_stack(me->def->body.iseq->cref_stack, me->klass, klass);
rb_add_method(klass, mid, VM_METHOD_TYPE_ISEQ, iseq, me->flag);
RB_GC_GUARD(newiseqval);
}