diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/nvim/eval/userfunc.c | 1 | ||||
| -rw-r--r-- | src/nvim/main.c | 13 | ||||
| -rw-r--r-- | src/nvim/ops.c | 8 | ||||
| -rw-r--r-- | src/nvim/testdir/test_registers.vim | 11 | ||||
| -rw-r--r-- | src/nvim/testdir/test_signals.vim | 140 | 
5 files changed, 171 insertions, 2 deletions
diff --git a/src/nvim/eval/userfunc.c b/src/nvim/eval/userfunc.c index 8daef00985..4e8b98d723 100644 --- a/src/nvim/eval/userfunc.c +++ b/src/nvim/eval/userfunc.c @@ -2484,6 +2484,7 @@ errret_2:    ga_clear_strings(&newlines);  ret_free:    xfree(skip_until); +  xfree(heredoc_trimmed);    xfree(line_to_free);    xfree(fudi.fd_newkey);    xfree(name); diff --git a/src/nvim/main.c b/src/nvim/main.c index ed055e76ba..41ae78e996 100644 --- a/src/nvim/main.c +++ b/src/nvim/main.c @@ -653,7 +653,18 @@ void getout(int exitval)          }        }      } -    apply_autocmds(EVENT_VIMLEAVEPRE, NULL, NULL, FALSE, curbuf); + +    int unblock = 0; +    // deathtrap() blocks autocommands, but we do want to trigger +    // VimLeavePre. +    if (is_autocmd_blocked()) { +      unblock_autocmds(); +      unblock++; +    } +    apply_autocmds(EVENT_VIMLEAVEPRE, NULL, NULL, false, curbuf); +    if (unblock) { +      block_autocmds(); +    }    }    if (p_shada && *p_shada != NUL) { diff --git a/src/nvim/ops.c b/src/nvim/ops.c index 37db1bb7e2..8fddb1b561 100644 --- a/src/nvim/ops.c +++ b/src/nvim/ops.c @@ -1197,7 +1197,13 @@ int insert_reg(        retval = FAIL;      } else {        for (size_t i = 0; i < reg->y_size; i++) { -        stuffescaped((const char *)reg->y_array[i], literally); +        if (regname == '-') { +          AppendCharToRedobuff(Ctrl_R); +          AppendCharToRedobuff(regname); +          do_put(regname, NULL, BACKWARD, 1L, PUT_CURSEND); +        } else { +          stuffescaped((const char *)reg->y_array[i], literally); +        }          // Insert a newline between lines and after last line if          // y_type is kMTLineWise.          if (reg->y_type == kMTLineWise || i < reg->y_size - 1) { diff --git a/src/nvim/testdir/test_registers.vim b/src/nvim/testdir/test_registers.vim index 19a7c6c9d0..24e02cb385 100644 --- a/src/nvim/testdir/test_registers.vim +++ b/src/nvim/testdir/test_registers.vim @@ -254,4 +254,15 @@ func Test_ve_blockpaste()    bwipe!  endfunc +func Test_insert_small_delete() +  new +  call setline(1, ['foo foobar bar']) +  call cursor(1,1) +  exe ":norm! ciw'\<C-R>-'" +  call assert_equal(getline(1), "'foo' foobar bar") +  exe ":norm! w.w." +  call assert_equal(getline(1), "'foo' 'foobar' 'bar'") +  bwipe! +endfunc +  " vim: shiftwidth=2 sts=2 expandtab diff --git a/src/nvim/testdir/test_signals.vim b/src/nvim/testdir/test_signals.vim new file mode 100644 index 0000000000..338c0d79ff --- /dev/null +++ b/src/nvim/testdir/test_signals.vim @@ -0,0 +1,140 @@ +" Test signal handling. + +source check.vim +source term_util.vim + +CheckUnix + +source shared.vim + +" Check whether a signal is available on this system. +func HasSignal(signal) +  let signals = system('kill -l') +  return signals =~# '\<' .. a:signal .. '\>' +endfunc + +" Test signal WINCH (window resize signal) +func Test_signal_WINCH() +  throw 'skipped: Nvim cannot avoid terminal resize' +  if has('gui_running') || !HasSignal('WINCH') +    return +  endif + +  " We do not actually want to change the size of the terminal. +  let old_WS = '' +  if exists('&t_WS') +    let old_WS = &t_WS +    let &t_WS = '' +  endif + +  let old_lines = &lines +  let old_columns = &columns +  let new_lines = &lines - 2 +  let new_columns = &columns - 2 + +  exe 'set lines=' .. new_lines +  exe 'set columns=' .. new_columns +  call assert_equal(new_lines, &lines) +  call assert_equal(new_columns, &columns) + +  " Send signal and wait for signal to be processed. +  " 'lines' and 'columns' should have been restored +  " after handing signal WINCH. +  exe 'silent !kill -s WINCH ' .. getpid() +  call WaitForAssert({-> assert_equal(old_lines, &lines)}) +  call assert_equal(old_columns, &columns) + +  if old_WS != '' +    let &t_WS = old_WS +  endif +endfunc + +" Test signal PWR, which should update the swap file. +func Test_signal_PWR() +  if !HasSignal('PWR') +    return +  endif + +  " Set a very large 'updatetime' and 'updatecount', so that we can be sure +  " that swap file is updated as a result of sending PWR signal, and not +  " because of exceeding 'updatetime' or 'updatecount' when changing buffer. +  set updatetime=100000 updatecount=100000 +  new Xtest_signal_PWR +  let swap_name = swapname('%') +  call setline(1, '123') +  preserve +  let swap_content = readfile(swap_name, 'b') + +  " Update the buffer and check that the swap file is not yet updated, +  " since we set 'updatetime' and 'updatecount' to large values. +  call setline(1, 'abc') +  call assert_equal(swap_content, readfile(swap_name, 'b')) + +  " Sending PWR signal should update the swap file. +  exe 'silent !kill -s PWR ' .. getpid() +  call WaitForAssert({-> assert_notequal(swap_content, readfile(swap_name, 'b'))}) + +  bwipe! +  set updatetime& updatecount& +endfunc + +" Test a deadly signal. +" +" There are several deadly signals: SISEGV, SIBUS, SIGTERM... +" Test uses signal SIGTERM as it does not create a core +" dump file unlike SIGSEGV, SIGBUS, etc. See "man 7 signals. +" +" Vim should exit with a deadly signal and unsaved changes +" should be recoverable from the swap file preserved as a +" result of the deadly signal handler. +func Test_deadly_signal_TERM() +  if !HasSignal('TERM') +    throw 'Skipped: TERM signal not supported' +  endif +  if !CanRunVimInTerminal() +    throw 'Skipped: cannot run vim in terminal' +  endif +  let cmd = GetVimCommand() +  if cmd =~ 'valgrind' +    throw 'Skipped: cannot test signal TERM with valgrind' +  endif + +  " If test fails once, it can leave temporary files and trying to rerun +  " the test would then fail again if they are not deleted first. +  call delete('.Xsig_TERM.swp') +  call delete('XsetupAucmd') +  call delete('XautoOut') +  let lines =<< trim END +    au VimLeave * call writefile(["VimLeave triggered"], "XautoOut", "as") +    au VimLeavePre * call writefile(["VimLeavePre triggered"], "XautoOut", "as") +  END +  call writefile(lines, 'XsetupAucmd') + +  let buf = RunVimInTerminal('-S XsetupAucmd Xsig_TERM', {'rows': 6}) +  let pid_vim = term_getjob(buf)->job_info().process + +  call term_sendkeys(buf, ":call setline(1, 'foo')\n") +  call WaitForAssert({-> assert_equal('foo', term_getline(buf, 1))}) + +  call assert_false(filereadable('Xsig_TERM')) +  exe 'silent !kill -s TERM '  .. pid_vim +  call WaitForAssert({-> assert_true(filereadable('.Xsig_TERM.swp'))}) + +  " Don't call StopVimInTerminal() as it expects job to be still running. +  call WaitForAssert({-> assert_equal("finished", term_getstatus(buf))}) + +  new +  silent recover .Xsig_TERM.swp +  call assert_equal(['foo'], getline(1, '$')) + +  let result = readfile('XautoOut') +  call assert_match('VimLeavePre triggered', result[0]) +  call assert_match('VimLeave triggered', result[1]) + +  %bwipe! +  call delete('.Xsig_TERM.swp') +  call delete('XsetupAucmd') +  call delete('XautoOut') +endfunc + +" vim: ts=8 sw=2 sts=2 tw=80 fdm=marker  | 
