diff options
| author | ko1 <ko1@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2007-05-21 04:46:51 +0000 |
|---|---|---|
| committer | ko1 <ko1@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2007-05-21 04:46:51 +0000 |
| commit | 151cab2bd7dcb0a983edb1488c59a82efdc28df7 (patch) | |
| tree | 5a1ed8dee97b8ee1c9a78aa83abd8f566246f1a3 /vm_macro.def | |
| parent | 79c54dec67f07004bedaa8ca079e68aeeb97267e (diff) | |
| download | ruby-151cab2bd7dcb0a983edb1488c59a82efdc28df7.tar.gz ruby-151cab2bd7dcb0a983edb1488c59a82efdc28df7.tar.xz ruby-151cab2bd7dcb0a983edb1488c59a82efdc28df7.zip | |
* compile.c, vm_macro.def: support tail call optimization
(on default, this feature is not enabled).
* iseq.c, compile.c, vm_opts.h: add "tailcall_optimization"
option.
* sample/test.rb (test_ok): fix to adjust tailcall stack layout.
* insns.def, vm.c, compile.c, yarvcore.c, yarvcore.h:
add opt_gt, opt_le instructions.
git-svn-id: http://svn.ruby-lang.org/repos/ruby/trunk@12304 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'vm_macro.def')
| -rw-r--r-- | vm_macro.def | 41 |
1 files changed, 23 insertions, 18 deletions
diff --git a/vm_macro.def b/vm_macro.def index dff6b98c9..adeaa52ac 100644 --- a/vm_macro.def +++ b/vm_macro.def @@ -198,33 +198,38 @@ MACRO macro_eval_invoke_func(niseqval, recv, klass, blockptr, num) rb_exc_raise(sysstack_error); } - for (i = 0; i < clear_local_size; i++) { - *sp++ = Qnil; - } + if (flag & VM_CALL_TAILCALL_BIT) { + /* copy arguments */ + VALUE *p_rsp, *p_sp; + reg_cfp = ++th->cfp; + p_rsp = th->cfp->sp; + + for (i=0; i < (sp - rsp); i++) { + p_rsp[i] = rsp[i]; + } + + sp -= rsp - p_rsp; - if (0 && (flag & VM_CALL_TAILCALL_BIT)) { - th->cfp++; - push_frame(th, niseq, FRAME_MAGIC_METHOD, - recv, (VALUE) blockptr, + for (i = 0; i < clear_local_size; i++) { + *sp++ = Qnil; + } + + push_frame(th, niseq, + FRAME_MAGIC_METHOD, recv, (VALUE) blockptr, niseq->iseq_encoded + opt_pc, sp, 0, 0); } - else if (0 && - (flag & VM_CALL_TAILRECURSION_BIT) && niseq == GET_ISEQ()) { - /* do nothing */ - GET_CFP()->self = recv; - SET_LFP(sp); - SET_DFP(sp); - *sp++ = (VALUE) blockptr; - reg_cfp->sp = sp; - reg_cfp->bp = sp; - SET_PC(niseq->iseq_encoded + opt_pc); - } else { + for (i = 0; i < clear_local_size; i++) { + *sp++ = Qnil; + } + push_frame(th, niseq, FRAME_MAGIC_METHOD, recv, (VALUE) blockptr, niseq->iseq_encoded + opt_pc, sp, 0, 0); + reg_cfp->sp = rsp; } + RESTORE_REGS(); } |
