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:
Damien George 2019-09-28 00:07:21 +10:00
parent 0096041c99
commit 82c494a97e
5 changed files with 127 additions and 39 deletions

42
py/vm.c
View File

@ -109,6 +109,21 @@
exc_sp--; /* pop back to previous exception handler */ \ 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 */ 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 #if MICROPY_PY_SYS_SETTRACE
#define FRAME_SETUP() do { \ #define FRAME_SETUP() do { \
@ -698,11 +713,15 @@ unwind_jump:;
while ((unum & 0x7f) > 0) { while ((unum & 0x7f) > 0) {
unum -= 1; unum -= 1;
assert(exc_sp >= exc_stack); 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: // Getting here the stack looks like:
// (..., X, dest_ip) // (..., X, dest_ip)
// where X is pointed to by exc_sp->val_sp and in the case // where X is pointed to by exc_sp->val_sp and in the case
// of a "with" block contains the context manager info. // 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 // We're going to run "finally" code as a coroutine
// (not calling it recursively). Set up a sentinel // (not calling it recursively). Set up a sentinel
// on the stack so it can return back to us when it is // 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 // The sentinel is the number of exception handlers left to
// unwind, which is a non-negative integer. // unwind, which is a non-negative integer.
PUSH(MP_OBJ_NEW_SMALL_INT(unum)); PUSH(MP_OBJ_NEW_SMALL_INT(unum));
ip = exc_sp->handler; // get exception handler byte code address ip = exc_sp->handler;
exc_sp--; // pop exception handler goto dispatch_loop;
goto dispatch_loop; // run the exception handler } else {
// Found a finally handler that is already active; cancel it.
CANCEL_ACTIVE_FINALLY(sp);
}
} }
POP_EXC_BLOCK(); POP_EXC_BLOCK();
} }
@ -740,9 +762,9 @@ unwind_jump:;
// if TOS is None, just pops it and continues // 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 integer, finishes coroutine and returns control to caller
// if TOS is an exception, reraises the exception // if TOS is an exception, reraises the exception
if (TOP() == mp_const_none) {
assert(exc_sp >= exc_stack); assert(exc_sp >= exc_stack);
POP_EXC_BLOCK(); POP_EXC_BLOCK();
if (TOP() == mp_const_none) {
sp--; sp--;
} else if (mp_obj_is_small_int(TOP())) { } else if (mp_obj_is_small_int(TOP())) {
// We finished "finally" coroutine and now dispatch back // We finished "finally" coroutine and now dispatch back
@ -1113,8 +1135,9 @@ unwind_jump:;
unwind_return: unwind_return:
// Search for and execute finally handlers that aren't already active // Search for and execute finally handlers that aren't already active
while (exc_sp >= exc_stack) { while (exc_sp >= exc_stack) {
if (MP_TAGPTR_TAG1(exc_sp->val_sp) && exc_sp->handler > ip) { if (MP_TAGPTR_TAG1(exc_sp->val_sp)) {
// Found a finally handler that isn't active. if (exc_sp->handler > ip) {
// Found a finally handler that isn't active; run it.
// Getting here the stack looks like: // Getting here the stack looks like:
// (..., X, [iter0, iter1, ...,] ret_val) // (..., X, [iter0, iter1, ...,] ret_val)
// where X is pointed to by exc_sp->val_sp and in the case // 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). // done (when WITH_CLEANUP or END_FINALLY reached).
PUSH(MP_OBJ_NEW_SMALL_INT(-1)); PUSH(MP_OBJ_NEW_SMALL_INT(-1));
ip = exc_sp->handler; ip = exc_sp->handler;
POP_EXC_BLOCK();
goto dispatch_loop; goto dispatch_loop;
} else {
// Found a finally handler that is already active; cancel it.
CANCEL_ACTIVE_FINALLY(sp);
}
} }
POP_EXC_BLOCK(); POP_EXC_BLOCK();
} }

View 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))

View 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))

View File

@ -0,0 +1,9 @@
4 0
continue
4 1
continue
4 2
continue
4 3
continue
None

View 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))