aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorzeertzjq <zeertzjq@outlook.com>2023-10-18 19:19:02 +0800
committerGitHub <noreply@github.com>2023-10-18 19:19:02 +0800
commitc49cfd89fdd32d670c484b0b677bc8647e891a12 (patch)
treeeef9986bd937fbd072ad6fc44eb489a6ba4ec994
parent831d662ac6756cab4fed6a9b394e68933b5fe325 (diff)
parent09fb243bdd0c2cde76e360bb98118932fda5d77b (diff)
downloadrneovim-c49cfd89fdd32d670c484b0b677bc8647e891a12.tar.gz
rneovim-c49cfd89fdd32d670c484b0b677bc8647e891a12.tar.bz2
rneovim-c49cfd89fdd32d670c484b0b677bc8647e891a12.zip
Merge pull request #25703 from zeertzjq/vim-9.0.2044
vim-patch:9.0.{2044,2045}
-rw-r--r--src/nvim/eval/userfunc.c15
-rw-r--r--test/old/testdir/runtest.vim22
-rw-r--r--test/old/testdir/test_user_func.vim26
3 files changed, 55 insertions, 8 deletions
diff --git a/src/nvim/eval/userfunc.c b/src/nvim/eval/userfunc.c
index ca98aad6bc..f789c53870 100644
--- a/src/nvim/eval/userfunc.c
+++ b/src/nvim/eval/userfunc.c
@@ -3296,8 +3296,23 @@ static void handle_defer_one(funccall_T *funccal)
char *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.
+ 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;
+
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;
+
tv_clear(&rettv);
xfree(name);
for (int i = dr->dr_argcount - 1; i >= 0; i--) {
diff --git a/test/old/testdir/runtest.vim b/test/old/testdir/runtest.vim
index 6be6fd0fd3..97ef96424a 100644
--- a/test/old/testdir/runtest.vim
+++ b/test/old/testdir/runtest.vim
@@ -231,6 +231,8 @@ func RunTheTest(test)
endtry
endif
+ let skipped = v:false
+
au VimLeavePre * call EarlyExit(g:testfunc)
if a:test =~ 'Test_nocatch_'
" Function handles errors itself. This avoids skipping commands after the
@@ -240,6 +242,7 @@ func RunTheTest(test)
if g:skipped_reason != ''
call add(s:messages, ' Skipped')
call add(s:skipped, 'SKIPPED ' . a:test . ': ' . g:skipped_reason)
+ let skipped = v:true
endif
else
try
@@ -247,6 +250,7 @@ func RunTheTest(test)
catch /^\cskipped/
call add(s:messages, ' Skipped')
call add(s:skipped, 'SKIPPED ' . a:test . ': ' . substitute(v:exception, '^\S*\s\+', '', ''))
+ let skipped = v:true
catch
call add(v:errors, 'Caught exception in ' . a:test . ': ' . v:exception . ' @ ' . v:throwpoint)
endtry
@@ -342,14 +346,16 @@ func RunTheTest(test)
endif
endwhile
- " Check if the test has left any swap files behind. Delete them before
- " running tests again, they might interfere.
- let swapfiles = s:GetSwapFileList()
- if len(swapfiles) > 0
- call add(s:messages, "Found swap files: " .. string(swapfiles))
- for name in swapfiles
- call delete(name)
- endfor
+ if !skipped
+ " Check if the test has left any swap files behind. Delete them before
+ " running tests again, they might interfere.
+ let swapfiles = s:GetSwapFileList()
+ if len(swapfiles) > 0
+ call add(s:messages, "Found swap files: " .. string(swapfiles))
+ for name in swapfiles
+ call delete(name)
+ endfor
+ endif
endif
endfunc
diff --git a/test/old/testdir/test_user_func.vim b/test/old/testdir/test_user_func.vim
index dc36ab98cb..ee1fd4ec5b 100644
--- a/test/old/testdir/test_user_func.vim
+++ b/test/old/testdir/test_user_func.vim
@@ -793,5 +793,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