diff options
| author | Yukihiro "Matz" Matsumoto <[email protected]> | 2017-03-11 16:32:29 +0900 |
|---|---|---|
| committer | Yukihiro "Matz" Matsumoto <[email protected]> | 2017-03-12 00:50:38 +0900 |
| commit | 26169f9e25788caf2781a92087f489e5e5fdc0c9 (patch) | |
| tree | 08fd028dc6857486eb470a057269d77b5fcd9356 /src/vm.c | |
| parent | 000c68da97ec0bfbd93e3969a2eef21081569a72 (diff) | |
| download | mruby-26169f9e25788caf2781a92087f489e5e5fdc0c9.tar.gz mruby-26169f9e25788caf2781a92087f489e5e5fdc0c9.zip | |
Enhance OP_RESCUE to take B operand fas matching exception; ref #3487
Diffstat (limited to 'src/vm.c')
| -rw-r--r-- | src/vm.c | 26 |
1 files changed, 25 insertions, 1 deletions
@@ -1083,7 +1083,31 @@ RETRY_TRY_BLOCK: CASE(OP_RESCUE) { /* A R(A) := exc; clear(exc) */ - SET_OBJ_VALUE(regs[GETARG_A(i)], mrb->exc); + /* B R(B) := matched (bool) */ + int b = GETARG_B(i); + mrb_value exc = mrb_obj_value(mrb->exc); + + if (b != 0) { + mrb_value e = regs[b]; + struct RClass *ec; + + switch (mrb_type(e)) { + case MRB_TT_CLASS: + case MRB_TT_MODULE: + break; + default: + mrb_raise(mrb, E_TYPE_ERROR, "class or module required for rescue clause"); + break; + } + ec = mrb_class_ptr(e); + if (mrb_obj_is_kind_of(mrb, exc, ec)) { + regs[b] = mrb_true_value(); + } + else { + regs[b] = mrb_false_value(); + } + } + regs[GETARG_A(i)] = exc; mrb->exc = 0; NEXT; } |
