diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/nvim/eval/userfunc.c | 19 | ||||
-rw-r--r-- | src/nvim/ex_eval.c | 29 | ||||
-rw-r--r-- | src/nvim/ex_eval_defs.h | 10 |
3 files changed, 46 insertions, 12 deletions
diff --git a/src/nvim/eval/userfunc.c b/src/nvim/eval/userfunc.c index f789c53870..ff86f74338 100644 --- a/src/nvim/eval/userfunc.c +++ b/src/nvim/eval/userfunc.c @@ -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); diff --git a/src/nvim/ex_eval.c b/src/nvim/ex_eval.c index c656f785c9..7412757726 100644 --- a/src/nvim/ex_eval.c +++ b/src/nvim/ex_eval.c @@ -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 diff --git a/src/nvim/ex_eval_defs.h b/src/nvim/ex_eval_defs.h index 6713cdb549..442e4581cc 100644 --- a/src/nvim/ex_eval_defs.h +++ b/src/nvim/ex_eval_defs.h @@ -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 |