mirror of
https://github.com/neovim/neovim.git
synced 2025-10-06 01:46:29 +00:00
vim-patch:9.0.2050: Vim9: crash with deferred function call and exception (#25715)
Problem: Vim9: crash with deferred function call and exception
Solution: Save and restore exception state
Crash when a deferred function is called after an exception and another
exception is thrown
closes: vim/vim#13376
closes: vim/vim#13377
c59c1e0d88
The change in check_due_timer() is N/A as Nvim calls timer callbacks on
the main loop.
Co-authored-by: Yegappan Lakshmanan <yegappan@yahoo.com>
This commit is contained in:
@@ -3297,21 +3297,16 @@ static void handle_defer_one(funccall_T *funccal)
|
||||
dr->dr_name = NULL;
|
||||
|
||||
// If the deferred function is called after an exception, then only the
|
||||
// first statement in the function will be executed. Save and restore
|
||||
// the try/catch/throw exception state.
|
||||
const int save_trylevel = trylevel;
|
||||
const bool save_did_throw = did_throw;
|
||||
const bool save_need_rethrow = need_rethrow;
|
||||
|
||||
trylevel = 0;
|
||||
did_throw = false;
|
||||
need_rethrow = false;
|
||||
// first statement in the function will be executed (because of the
|
||||
// exception). So save and restore the try/catch/throw exception
|
||||
// state.
|
||||
exception_state_T estate;
|
||||
exception_state_save(&estate);
|
||||
exception_state_clear();
|
||||
|
||||
call_func(name, -1, &rettv, dr->dr_argcount, dr->dr_argvars, &funcexe);
|
||||
|
||||
trylevel = save_trylevel;
|
||||
did_throw = save_did_throw;
|
||||
need_rethrow = save_need_rethrow;
|
||||
exception_state_restore(&estate);
|
||||
|
||||
tv_clear(&rettv);
|
||||
xfree(name);
|
||||
|
@@ -661,6 +661,35 @@ static void finish_exception(except_T *excp)
|
||||
discard_exception(excp, true);
|
||||
}
|
||||
|
||||
/// Save the current exception state in "estate"
|
||||
void exception_state_save(exception_state_T *estate)
|
||||
{
|
||||
estate->estate_current_exception = current_exception;
|
||||
estate->estate_did_throw = did_throw;
|
||||
estate->estate_need_rethrow = need_rethrow;
|
||||
estate->estate_trylevel = trylevel;
|
||||
}
|
||||
|
||||
/// Restore the current exception state from "estate"
|
||||
void exception_state_restore(exception_state_T *estate)
|
||||
{
|
||||
if (current_exception == NULL) {
|
||||
current_exception = estate->estate_current_exception;
|
||||
}
|
||||
did_throw |= estate->estate_did_throw;
|
||||
need_rethrow |= estate->estate_need_rethrow;
|
||||
trylevel |= estate->estate_trylevel;
|
||||
}
|
||||
|
||||
/// Clear the current exception state
|
||||
void exception_state_clear(void)
|
||||
{
|
||||
current_exception = NULL;
|
||||
did_throw = false;
|
||||
need_rethrow = false;
|
||||
trylevel = 0;
|
||||
}
|
||||
|
||||
// Flags specifying the message displayed by report_pending.
|
||||
#define RP_MAKE 0
|
||||
#define RP_RESUME 1
|
||||
|
@@ -118,4 +118,14 @@ struct cleanup_stuff {
|
||||
except_T *exception; ///< exception value
|
||||
};
|
||||
|
||||
/// Exception state that is saved and restored when calling timer callback
|
||||
/// functions and deferred functions.
|
||||
typedef struct exception_state_S exception_state_T;
|
||||
struct exception_state_S {
|
||||
except_T *estate_current_exception;
|
||||
bool estate_did_throw;
|
||||
bool estate_need_rethrow;
|
||||
int estate_trylevel;
|
||||
};
|
||||
|
||||
#endif // NVIM_EX_EVAL_DEFS_H
|
||||
|
Reference in New Issue
Block a user