aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/nvim/ex_docmd.c19
-rw-r--r--src/nvim/memline.c33
-rw-r--r--src/nvim/misc1.c4
-rw-r--r--src/nvim/testdir/test_search.vim9
4 files changed, 44 insertions, 21 deletions
diff --git a/src/nvim/ex_docmd.c b/src/nvim/ex_docmd.c
index ac5de0733b..e4ab690bb9 100644
--- a/src/nvim/ex_docmd.c
+++ b/src/nvim/ex_docmd.c
@@ -3664,17 +3664,18 @@ static linenr_T get_address(exarg_T *eap,
*/
if (lnum != MAXLNUM)
curwin->w_cursor.lnum = lnum;
- /*
- * Start a forward search at the end of the line.
- * Start a backward search at the start of the line.
- * This makes sure we never match in the current
- * line, and can match anywhere in the
- * next/previous line.
- */
- if (c == '/')
+
+ // Start a forward search at the end of the line (unless
+ // before the first line).
+ // Start a backward search at the start of the line.
+ // This makes sure we never match in the current
+ // line, and can match anywhere in the
+ // next/previous line.
+ if (c == '/' && curwin->w_cursor.lnum > 0) {
curwin->w_cursor.col = MAXCOL;
- else
+ } else {
curwin->w_cursor.col = 0;
+ }
searchcmdlen = 0;
if (!do_search(NULL, c, cmd, 1L,
SEARCH_HIS | SEARCH_MSG, NULL)) {
diff --git a/src/nvim/memline.c b/src/nvim/memline.c
index ec9996810f..e84b8d623d 100644
--- a/src/nvim/memline.c
+++ b/src/nvim/memline.c
@@ -3999,18 +3999,15 @@ void goto_byte(long cnt)
/// Return 0 otherwise.
int inc(pos_T *lp)
{
- char_u *p = ml_get_pos(lp);
-
- if (*p != NUL) { // still within line, move to next char (may be NUL)
- if (has_mbyte) {
- int l = (*mb_ptr2len)(p);
+ // when searching position may be set to end of a line
+ if (lp->col != MAXCOL) {
+ const char_u *const p = ml_get_pos(lp);
+ if (*p != NUL) { // still within line, move to next char (may be NUL)
+ const int l = utfc_ptr2len(p);
lp->col += l;
- return (p[l] != NUL) ? 0 : 2;
+ return ((p[l] != NUL) ? 0 : 2);
}
- lp->col++;
- lp->coladd = 0;
- return (p[1] != NUL) ? 0 : 2;
}
if (lp->lnum != curbuf->b_ml.ml_line_count) { // there is a next line
lp->col = 0;
@@ -4035,20 +4032,32 @@ int incl(pos_T *lp)
int dec(pos_T *lp)
{
lp->coladd = 0;
- if (lp->col > 0) { // still within line
+ if (lp->col == MAXCOL) {
+ // past end of line
+ char_u *p = ml_get(lp->lnum);
+ lp->col = (colnr_T)STRLEN(p);
+ lp->col -= utf_head_off(p, p + lp->col);
+ return 0;
+ }
+
+ if (lp->col > 0) {
+ // still within line
lp->col--;
char_u *p = ml_get(lp->lnum);
lp->col -= utf_head_off(p, p + lp->col);
return 0;
}
- if (lp->lnum > 1) { // there is a prior line
+ if (lp->lnum > 1) {
+ // there is a prior line
lp->lnum--;
char_u *p = ml_get(lp->lnum);
lp->col = (colnr_T)STRLEN(p);
lp->col -= utf_head_off(p, p + lp->col);
return 1;
}
- return -1; // at start of file
+
+ // at start of file
+ return -1;
}
/// Same as dec(), but skip NUL at the end of non-empty lines.
diff --git a/src/nvim/misc1.c b/src/nvim/misc1.c
index ad0a8d409f..99dc29f119 100644
--- a/src/nvim/misc1.c
+++ b/src/nvim/misc1.c
@@ -1752,6 +1752,10 @@ del_lines (
int gchar_pos(pos_T *pos)
{
+ // When searching columns is sometimes put at the end of a line.
+ if (pos->col == MAXCOL) {
+ return NUL;
+ }
return utf_ptr2char(ml_get_pos(pos));
}
diff --git a/src/nvim/testdir/test_search.vim b/src/nvim/testdir/test_search.vim
index 7663c9e283..ecd840d40c 100644
--- a/src/nvim/testdir/test_search.vim
+++ b/src/nvim/testdir/test_search.vim
@@ -480,3 +480,12 @@ func Test_look_behind()
call search(getline("."))
bwipe!
endfunc
+
+func Test_search_sentence()
+ new
+ " this used to cause a crash
+ call assert_fails("/\\%')", 'E486')
+ call assert_fails("/", 'E486')
+ /\%'(
+ /
+endfunc