mirror of
https://github.com/vim/vim.git
synced 2025-09-25 03:54:15 -04:00
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:
committed by
Christian Brabandt
parent
5036e69852
commit
0672595fd5
@@ -870,5 +870,31 @@ func Test_defer_wrong_arguments()
|
|||||||
call v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
|
call v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
|
||||||
endfunc
|
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
|
" vim: shiftwidth=2 sts=2 expandtab
|
||||||
|
@@ -4686,6 +4686,35 @@ def Test_refer_funcref_instr_after_realloc()
|
|||||||
v9.CheckScriptSuccess(lines)
|
v9.CheckScriptSuccess(lines)
|
||||||
enddef
|
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.
|
" Keep this last, it messes up highlighting.
|
||||||
def Test_substitute_cmd()
|
def Test_substitute_cmd()
|
||||||
new
|
new
|
||||||
|
@@ -6251,8 +6251,23 @@ handle_defer_one(funccall_T *funccal)
|
|||||||
char_u *name = dr->dr_name;
|
char_u *name = dr->dr_name;
|
||||||
dr->dr_name = NULL;
|
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);
|
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);
|
clear_tv(&rettv);
|
||||||
vim_free(name);
|
vim_free(name);
|
||||||
for (int i = dr->dr_argcount - 1; i >= 0; --i)
|
for (int i = dr->dr_argcount - 1; i >= 0; --i)
|
||||||
|
@@ -704,6 +704,8 @@ static char *(features[]) =
|
|||||||
|
|
||||||
static int included_patches[] =
|
static int included_patches[] =
|
||||||
{ /* Add new patch number below this line */
|
{ /* Add new patch number below this line */
|
||||||
|
/**/
|
||||||
|
2044,
|
||||||
/**/
|
/**/
|
||||||
2043,
|
2043,
|
||||||
/**/
|
/**/
|
||||||
|
@@ -1140,8 +1140,23 @@ invoke_defer_funcs(ectx_T *ectx)
|
|||||||
char_u *name = functv->vval.v_string;
|
char_u *name = functv->vval.v_string;
|
||||||
functv->vval.v_string = NULL;
|
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);
|
(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);
|
clear_tv(&rettv);
|
||||||
vim_free(name);
|
vim_free(name);
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user