diff options
author | Bram Moolenaar <Bram@vim.org> | 2022-01-17 17:30:21 +0000 |
---|---|---|
committer | Bram Moolenaar <Bram@vim.org> | 2022-01-17 17:30:21 +0000 |
commit | 9f8c304c8a390ade133bac29963dc8e56ab14cbc (patch) | |
tree | 3878d49658f574c119bca93b518cccd316eb693b /src | |
parent | e8741a73e20cd6043dd1610fb4c22ae9d132eb56 (diff) | |
download | vim-git-9f8c304c8a390ade133bac29963dc8e56ab14cbc.tar.gz |
patch 8.2.4120: block insert goes over the end of the linev8.2.4120
Problem: Block insert goes over the end of the line.
Solution: Handle invalid byte better. Fix inserting the wrong text.
Diffstat (limited to 'src')
-rw-r--r-- | src/ops.c | 40 | ||||
-rw-r--r-- | src/testdir/test_visual.vim | 9 | ||||
-rw-r--r-- | src/version.c | 2 |
3 files changed, 37 insertions, 14 deletions
@@ -536,24 +536,29 @@ block_insert( if (b_insert) { off = (*mb_head_off)(oldp, oldp + offset + spaces); + spaces -= off; + count -= off; } else { - off = (*mb_off_next)(oldp, oldp + offset); - offset += off; + // spaces fill the gap, the character that's at the edge moves + // right + off = (*mb_head_off)(oldp, oldp + offset); + offset -= off; } - spaces -= off; - count -= off; } if (spaces < 0) // can happen when the cursor was moved spaces = 0; - newp = alloc(STRLEN(oldp) + s_len + count + 1); + // Make sure the allocated size matches what is actually copied below. + newp = alloc(STRLEN(oldp) + spaces + s_len + + (spaces > 0 && !bdp->is_short ? ts_val - spaces : 0) + + count + 1); if (newp == NULL) continue; // copy up to shifted part - mch_memmove(newp, oldp, (size_t)(offset)); + mch_memmove(newp, oldp, (size_t)offset); oldp += offset; // insert pre-padding @@ -564,14 +569,21 @@ block_insert( mch_memmove(newp + startcol, s, (size_t)s_len); offset += s_len; - if (spaces && !bdp->is_short) + if (spaces > 0 && !bdp->is_short) { - // insert post-padding - vim_memset(newp + offset + spaces, ' ', (size_t)(ts_val - spaces)); - // We're splitting a TAB, don't copy it. - oldp++; - // We allowed for that TAB, remember this now - count++; + if (*oldp == TAB) + { + // insert post-padding + vim_memset(newp + offset + spaces, ' ', + (size_t)(ts_val - spaces)); + // we're splitting a TAB, don't copy it + oldp++; + // We allowed for that TAB, remember this now + count++; + } + else + // Not a TAB, no extra spaces + count = spaces; } if (spaces > 0) @@ -1598,7 +1610,7 @@ op_insert(oparg_T *oap, long count1) oap->start_vcol = t; } else if (oap->op_type == OP_APPEND - && oap->end.col + oap->end.coladd + && oap->start.col + oap->start.coladd >= curbuf->b_op_start_orig.col + curbuf->b_op_start_orig.coladd) { diff --git a/src/testdir/test_visual.vim b/src/testdir/test_visual.vim index e40be5ded..9ca0fa9c8 100644 --- a/src/testdir/test_visual.vim +++ b/src/testdir/test_visual.vim @@ -1278,6 +1278,15 @@ func Test_visual_block_ctrl_w_f() au! BufNew endfunc +func Test_visual_block_append_invalid_char() + " this was going over the end of the line + new + call setline(1, [' let xxx', 'xxxxx', 'xxxxxxxxxxx']) + exe "normal 0\<C-V>jjA-\<Esc>" + call assert_equal([' - let xxx', 'xxxxx -', 'xxxxxxxx-xxx'], getline(1, 3)) + bwipe! +endfunc + func Test_visual_reselect_with_count() " this was causing an illegal memory access let lines =<< trim END diff --git a/src/version.c b/src/version.c index 01b8e5d3d..6fda77e93 100644 --- a/src/version.c +++ b/src/version.c @@ -751,6 +751,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ /**/ + 4120, +/**/ 4119, /**/ 4118, |