From c2f3a494ea3473d2e42884545ee23e532eb83bd3 Mon Sep 17 00:00:00 2001 From: "Yukihiro \"Matz\" Matsumoto" Date: Mon, 22 Apr 2013 16:32:46 +0900 Subject: ensure clause should be executed only once; close #1185 --- src/vm.c | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/vm.c b/src/vm.c index d9f2d6d8c..ea3b38adf 100644 --- a/src/vm.c +++ b/src/vm.c @@ -237,6 +237,7 @@ ecall(mrb_state *mrb, int i) struct RObject *exc; p = mrb->ensure[i]; + if (!p) return; ci = cipush(mrb); ci->stackidx = mrb->stack - mrb->stbase; ci->mid = ci[-1].mid; @@ -248,6 +249,7 @@ ecall(mrb_state *mrb, int i) mrb->stack = mrb->stack + ci[-1].nregs; exc = mrb->exc; mrb->exc = 0; mrb_run(mrb, p, *self); + mrb->ensure[i] = NULL; if (!mrb->exc) mrb->exc = exc; } @@ -811,8 +813,7 @@ mrb_run(mrb_state *mrb, struct RProc *proc, mrb_value self) int a = GETARG_A(i); for (n=0; nci->eidx-1); - --mrb->ci->eidx; + ecall(mrb, --mrb->ci->eidx); } mrb_gc_arena_restore(mrb, ai); NEXT; @@ -1217,6 +1218,9 @@ mrb_run(mrb_state *mrb, struct RProc *proc, mrb_value self) if (ci->ridx == 0) goto L_STOP; goto L_RESCUE; } + while (eidx > ci[-1].eidx) { + ecall(mrb, --eidx); + } while (ci[0].ridx == ci[-1].ridx) { cipop(mrb); ci = mrb->ci; @@ -1225,7 +1229,7 @@ mrb_run(mrb_state *mrb, struct RProc *proc, mrb_value self) mrb->jmp = prev_jmp; longjmp(*(jmp_buf*)mrb->jmp, 1); } - while (eidx > mrb->ci->eidx) { + while (eidx > ci->eidx) { ecall(mrb, --eidx); } if (ci == mrb->cibase) { -- cgit v1.2.3