patch 9.0.2044: Vim9: exceptions confuse defered functions

Problem:  Vim9: exceptions confuse defered functions
Solution: save and restore exception state when calling defered
          functions

closes: #13364
closes: #13372

Signed-off-by: Christian Brabandt <cb@256bit.org>
Co-authored-by: Yegappan Lakshmanan <yegappan@yahoo.com>
This commit is contained in:
Yegappan Lakshmanan
2023-10-18 11:47:37 +02:00
committed by Christian Brabandt
parent 5036e69852
commit 0672595fd5
5 changed files with 87 additions and 0 deletions

View File

@ -870,5 +870,31 @@ func Test_defer_wrong_arguments()
call v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
endfunc
" Test for calling a deferred function after an exception
func Test_defer_after_exception()
let g:callTrace = []
func Defer()
let g:callTrace += ['a']
let g:callTrace += ['b']
let g:callTrace += ['c']
let g:callTrace += ['d']
endfunc
func Foo()
defer Defer()
throw "TestException"
endfunc
try
call Foo()
catch /TestException/
let g:callTrace += ['e']
endtry
call assert_equal(['a', 'b', 'c', 'd', 'e'], g:callTrace)
delfunc Defer
delfunc Foo
unlet g:callTrace
endfunc
" vim: shiftwidth=2 sts=2 expandtab

View File

@ -4686,6 +4686,35 @@ def Test_refer_funcref_instr_after_realloc()
v9.CheckScriptSuccess(lines)
enddef
" Test for calling a deferred function after an exception
def Test_defer_after_exception()
var lines =<< trim END
vim9script
var callTrace: list<string> = []
def Defer()
callTrace += ['a']
callTrace += ['b']
callTrace += ['c']
callTrace += ['d']
enddef
def Foo()
defer Defer()
throw "TestException"
enddef
try
Foo()
catch /TestException/
callTrace += ['e']
endtry
assert_equal(['a', 'b', 'c', 'd', 'e'], callTrace)
END
v9.CheckScriptSuccess(lines)
enddef
" Keep this last, it messes up highlighting.
def Test_substitute_cmd()
new

View File

@ -6251,8 +6251,23 @@ handle_defer_one(funccall_T *funccal)
char_u *name = dr->dr_name;
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.
int save_trylevel = trylevel;
int save_did_throw = did_throw;
int save_need_rethrow = need_rethrow;
trylevel = 0;
did_throw = FALSE;
need_rethrow = FALSE;
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;
clear_tv(&rettv);
vim_free(name);
for (int i = dr->dr_argcount - 1; i >= 0; --i)

View File

@ -704,6 +704,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
2044,
/**/
2043,
/**/

View File

@ -1140,8 +1140,23 @@ invoke_defer_funcs(ectx_T *ectx)
char_u *name = functv->vval.v_string;
functv->vval.v_string = 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.
int save_trylevel = trylevel;
int save_did_throw = did_throw;
int save_need_rethrow = need_rethrow;
trylevel = 0;
did_throw = FALSE;
need_rethrow = FALSE;
(void)call_func(name, -1, &rettv, argcount, argvars, &funcexe);
trylevel = save_trylevel;
did_throw = save_did_throw;
need_rethrow = save_need_rethrow;
clear_tv(&rettv);
vim_free(name);
}