diff options
author | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2016-03-02 07:28:50 +0000 |
---|---|---|
committer | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2016-03-02 07:28:50 +0000 |
commit | 0534b970bc83814e77ea4bd8ba9b2d31fefc9b1e (patch) | |
tree | ab3ec937d674ff708934cc24a89effb6a80578da | |
parent | be8005b977a2044534cef7c468a07fbc34f6b3d2 (diff) | |
download | ruby-0534b970bc83814e77ea4bd8ba9b2d31fefc9b1e.tar.gz |
vm_eval.c: call method_missing by method entry
* vm_eval.c (method_missing): call by found method entry and get
rid of searching the same method entry twice.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53984 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r-- | ChangeLog | 5 | ||||
-rw-r--r-- | vm_eval.c | 15 |
2 files changed, 14 insertions, 6 deletions
@@ -1,4 +1,7 @@ -Wed Mar 2 16:18:30 2016 Nobuyoshi Nakada <nobu@ruby-lang.org> +Wed Mar 2 16:28:48 2016 Nobuyoshi Nakada <nobu@ruby-lang.org> + + * vm_eval.c (method_missing): call by found method entry and get + rid of searching the same method entry twice. * vm_eval.c (vm_call0_body): calling method_missing method is method_missing(). @@ -747,26 +747,31 @@ raise_method_missing(rb_thread_t *th, int argc, const VALUE *argv, VALUE obj, static inline VALUE method_missing(VALUE obj, ID id, int argc, const VALUE *argv, enum method_missing_reason call_status) { - VALUE *nargv, result, work; + VALUE *nargv, result, work, klass; rb_thread_t *th = GET_THREAD(); const rb_block_t *blockptr = th->passed_block; + const rb_callable_method_entry_t *me; th->method_missing_reason = call_status; th->passed_block = 0; if (id == idMethodMissing) { + missing: raise_method_missing(th, argc, argv, obj, call_status | MISSING_MISSING); } nargv = ALLOCV_N(VALUE, work, argc + 1); nargv[0] = ID2SYM(id); MEMCPY(nargv + 1, argv, VALUE, argc); + ++argc; + argv = nargv; - if (rb_method_basic_definition_p(CLASS_OF(obj) , idMethodMissing)) { - raise_method_missing(th, argc+1, nargv, obj, call_status | MISSING_MISSING); - } + klass = CLASS_OF(obj); + if (!klass) goto missing; + me = rb_callable_method_entry(klass, idMethodMissing); + if (!me || METHOD_ENTRY_BASIC(me)) goto missing; th->passed_block = blockptr; - result = rb_funcall2(obj, idMethodMissing, argc + 1, nargv); + result = vm_call0(th, obj, idMethodMissing, argc, argv, me); if (work) ALLOCV_END(work); return result; } |