py/vm: Fix handling of unwind jump out of active finally.
Prior to this commit, when unwinding through an active finally the stack was not being correctly popped/folded, which resulting in the VM crashing for complicated unwinding of nested finallys. This should be fixed with this commit, and more tests for return/break/ continue within a finally have been added to exercise this.
This commit is contained in:
parent
0096041c99
commit
82c494a97e
42
py/vm.c
42
py/vm.c
@ -109,6 +109,21 @@
|
||||
exc_sp--; /* pop back to previous exception handler */ \
|
||||
CLEAR_SYS_EXC_INFO() /* just clear sys.exc_info(), not compliant, but it shouldn't be used in 1st place */
|
||||
|
||||
#define CANCEL_ACTIVE_FINALLY(sp) do { \
|
||||
if (mp_obj_is_small_int(sp[-1])) { \
|
||||
/* Stack: (..., prev_dest_ip, prev_cause, dest_ip) */ \
|
||||
/* Cancel the unwind through the previous finally, replace with current one */ \
|
||||
sp[-2] = sp[0]; \
|
||||
sp -= 2; \
|
||||
} else { \
|
||||
assert(sp[-1] == mp_const_none || mp_obj_is_exception_instance(sp[-1])); \
|
||||
/* Stack: (..., None/exception, dest_ip) */ \
|
||||
/* Silence the finally's exception value (may be None or an exception) */ \
|
||||
sp[-1] = sp[0]; \
|
||||
--sp; \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#if MICROPY_PY_SYS_SETTRACE
|
||||
|
||||
#define FRAME_SETUP() do { \
|
||||
@ -698,11 +713,15 @@ unwind_jump:;
|
||||
while ((unum & 0x7f) > 0) {
|
||||
unum -= 1;
|
||||
assert(exc_sp >= exc_stack);
|
||||
if (MP_TAGPTR_TAG1(exc_sp->val_sp) && exc_sp->handler > ip) {
|
||||
|
||||
if (MP_TAGPTR_TAG1(exc_sp->val_sp)) {
|
||||
if (exc_sp->handler > ip) {
|
||||
// Found a finally handler that isn't active; run it.
|
||||
// Getting here the stack looks like:
|
||||
// (..., X, dest_ip)
|
||||
// where X is pointed to by exc_sp->val_sp and in the case
|
||||
// of a "with" block contains the context manager info.
|
||||
assert(&sp[-1] == MP_TAGPTR_PTR(exc_sp->val_sp));
|
||||
// We're going to run "finally" code as a coroutine
|
||||
// (not calling it recursively). Set up a sentinel
|
||||
// on the stack so it can return back to us when it is
|
||||
@ -710,9 +729,12 @@ unwind_jump:;
|
||||
// The sentinel is the number of exception handlers left to
|
||||
// unwind, which is a non-negative integer.
|
||||
PUSH(MP_OBJ_NEW_SMALL_INT(unum));
|
||||
ip = exc_sp->handler; // get exception handler byte code address
|
||||
exc_sp--; // pop exception handler
|
||||
goto dispatch_loop; // run the exception handler
|
||||
ip = exc_sp->handler;
|
||||
goto dispatch_loop;
|
||||
} else {
|
||||
// Found a finally handler that is already active; cancel it.
|
||||
CANCEL_ACTIVE_FINALLY(sp);
|
||||
}
|
||||
}
|
||||
POP_EXC_BLOCK();
|
||||
}
|
||||
@ -740,9 +762,9 @@ unwind_jump:;
|
||||
// if TOS is None, just pops it and continues
|
||||
// if TOS is an integer, finishes coroutine and returns control to caller
|
||||
// if TOS is an exception, reraises the exception
|
||||
if (TOP() == mp_const_none) {
|
||||
assert(exc_sp >= exc_stack);
|
||||
POP_EXC_BLOCK();
|
||||
if (TOP() == mp_const_none) {
|
||||
sp--;
|
||||
} else if (mp_obj_is_small_int(TOP())) {
|
||||
// We finished "finally" coroutine and now dispatch back
|
||||
@ -1113,8 +1135,9 @@ unwind_jump:;
|
||||
unwind_return:
|
||||
// Search for and execute finally handlers that aren't already active
|
||||
while (exc_sp >= exc_stack) {
|
||||
if (MP_TAGPTR_TAG1(exc_sp->val_sp) && exc_sp->handler > ip) {
|
||||
// Found a finally handler that isn't active.
|
||||
if (MP_TAGPTR_TAG1(exc_sp->val_sp)) {
|
||||
if (exc_sp->handler > ip) {
|
||||
// Found a finally handler that isn't active; run it.
|
||||
// Getting here the stack looks like:
|
||||
// (..., X, [iter0, iter1, ...,] ret_val)
|
||||
// where X is pointed to by exc_sp->val_sp and in the case
|
||||
@ -1133,8 +1156,11 @@ unwind_return:
|
||||
// done (when WITH_CLEANUP or END_FINALLY reached).
|
||||
PUSH(MP_OBJ_NEW_SMALL_INT(-1));
|
||||
ip = exc_sp->handler;
|
||||
POP_EXC_BLOCK();
|
||||
goto dispatch_loop;
|
||||
} else {
|
||||
// Found a finally handler that is already active; cancel it.
|
||||
CANCEL_ACTIVE_FINALLY(sp);
|
||||
}
|
||||
}
|
||||
POP_EXC_BLOCK();
|
||||
}
|
||||
|
19
tests/basics/try_finally_break2.py
Normal file
19
tests/basics/try_finally_break2.py
Normal file
@ -0,0 +1,19 @@
|
||||
def foo(x):
|
||||
for i in range(x):
|
||||
for j in range(x):
|
||||
try:
|
||||
print(x, i, j, 1)
|
||||
finally:
|
||||
try:
|
||||
try:
|
||||
print(x, i, j, 2)
|
||||
finally:
|
||||
try:
|
||||
1 / 0
|
||||
finally:
|
||||
print(x, i, j, 3)
|
||||
break
|
||||
finally:
|
||||
print(x, i, j, 4)
|
||||
break
|
||||
print(foo(4))
|
17
tests/basics/try_finally_continue.py
Normal file
17
tests/basics/try_finally_continue.py
Normal file
@ -0,0 +1,17 @@
|
||||
def foo(x):
|
||||
for i in range(x):
|
||||
try:
|
||||
pass
|
||||
finally:
|
||||
try:
|
||||
try:
|
||||
print(x, i)
|
||||
finally:
|
||||
try:
|
||||
1 / 0
|
||||
finally:
|
||||
return 42
|
||||
finally:
|
||||
print('continue')
|
||||
continue
|
||||
print(foo(4))
|
9
tests/basics/try_finally_continue.py.exp
Normal file
9
tests/basics/try_finally_continue.py.exp
Normal file
@ -0,0 +1,9 @@
|
||||
4 0
|
||||
continue
|
||||
4 1
|
||||
continue
|
||||
4 2
|
||||
continue
|
||||
4 3
|
||||
continue
|
||||
None
|
17
tests/basics/try_finally_return5.py
Normal file
17
tests/basics/try_finally_return5.py
Normal file
@ -0,0 +1,17 @@
|
||||
def foo(x):
|
||||
for i in range(x):
|
||||
try:
|
||||
pass
|
||||
finally:
|
||||
try:
|
||||
try:
|
||||
print(x, i)
|
||||
finally:
|
||||
try:
|
||||
1 / 0
|
||||
finally:
|
||||
return 42
|
||||
finally:
|
||||
print('return')
|
||||
return 43
|
||||
print(foo(4))
|
Loading…
x
Reference in New Issue
Block a user