0
0
mirror of https://github.com/vim/vim.git synced 2025-09-25 03:54:15 -04:00

patch 8.1.2269: tags file with very long line stops using binary search

Problem:    Tags file with very long line stops using binary search.
Solution:   Reallocate the buffer if needed.
This commit is contained in:
Bram Moolenaar
2019-11-07 23:08:42 +01:00
parent 3d2a47c782
commit dc9ef26845
3 changed files with 41 additions and 25 deletions

View File

@@ -1937,7 +1937,7 @@ find_tags(
*/ */
else if (state == TS_SKIP_BACK) else if (state == TS_SKIP_BACK)
{ {
search_info.curr_offset -= LSIZE * 2; search_info.curr_offset -= lbuf_size * 2;
if (search_info.curr_offset < 0) if (search_info.curr_offset < 0)
{ {
search_info.curr_offset = 0; search_info.curr_offset = 0;
@@ -1955,7 +1955,7 @@ find_tags(
/* Adjust the search file offset to the correct position */ /* Adjust the search file offset to the correct position */
search_info.curr_offset_used = search_info.curr_offset; search_info.curr_offset_used = search_info.curr_offset;
vim_fseek(fp, search_info.curr_offset, SEEK_SET); vim_fseek(fp, search_info.curr_offset, SEEK_SET);
eof = vim_fgets(lbuf, LSIZE, fp); eof = vim_fgets(lbuf, lbuf_size, fp);
if (!eof && search_info.curr_offset != 0) if (!eof && search_info.curr_offset != 0)
{ {
/* The explicit cast is to work around a bug in gcc 3.4.2 /* The explicit cast is to work around a bug in gcc 3.4.2
@@ -1967,13 +1967,13 @@ find_tags(
vim_fseek(fp, search_info.low_offset, SEEK_SET); vim_fseek(fp, search_info.low_offset, SEEK_SET);
search_info.curr_offset = search_info.low_offset; search_info.curr_offset = search_info.low_offset;
} }
eof = vim_fgets(lbuf, LSIZE, fp); eof = vim_fgets(lbuf, lbuf_size, fp);
} }
/* skip empty and blank lines */ /* skip empty and blank lines */
while (!eof && vim_isblankline(lbuf)) while (!eof && vim_isblankline(lbuf))
{ {
search_info.curr_offset = vim_ftell(fp); search_info.curr_offset = vim_ftell(fp);
eof = vim_fgets(lbuf, LSIZE, fp); eof = vim_fgets(lbuf, lbuf_size, fp);
} }
if (eof) if (eof)
{ {
@@ -1996,10 +1996,10 @@ find_tags(
{ {
#ifdef FEAT_CSCOPE #ifdef FEAT_CSCOPE
if (use_cscope) if (use_cscope)
eof = cs_fgets(lbuf, LSIZE); eof = cs_fgets(lbuf, lbuf_size);
else else
#endif #endif
eof = vim_fgets(lbuf, LSIZE, fp); eof = vim_fgets(lbuf, lbuf_size, fp);
} while (!eof && vim_isblankline(lbuf)); } while (!eof && vim_isblankline(lbuf));
if (eof) if (eof)
@@ -2230,28 +2230,21 @@ parse_line:
// When the line is too long the NUL will not be in the // When the line is too long the NUL will not be in the
// last-but-one byte (see vim_fgets()). // last-but-one byte (see vim_fgets()).
// Has been reported for Mozilla JS with extremely long names. // Has been reported for Mozilla JS with extremely long names.
// In that case we can't parse it and we ignore the line. // In that case we need to increase lbuf_size.
if (lbuf[LSIZE - 2] != NUL if (lbuf[lbuf_size - 2] != NUL
#ifdef FEAT_CSCOPE #ifdef FEAT_CSCOPE
&& !use_cscope && !use_cscope
#endif #endif
) )
{ {
if (p_verbose >= 5) lbuf_size *= 2;
{ vim_free(lbuf);
verbose_enter(); lbuf = alloc(lbuf_size);
msg(_("Ignoring long line in tags file")); if (lbuf == NULL)
verbose_leave(); goto findtag_end;
} // this will try the same thing again, make sure the offset is
#ifdef FEAT_TAG_BINS // different
if (state != TS_LINEAR) search_info.curr_offset = 0;
{
// Avoid getting stuck.
linear = TRUE;
state = TS_LINEAR;
vim_fseek(fp, search_info.low_offset, SEEK_SET);
}
#endif
continue; continue;
} }
@@ -3367,6 +3360,8 @@ jumpto_tag(
break; break;
#endif #endif
*pbuf_end++ = *str++; *pbuf_end++ = *str++;
if (pbuf_end - pbuf + 1 >= LSIZE)
break;
} }
*pbuf_end = NUL; *pbuf_end = NUL;

View File

@@ -459,7 +459,8 @@ func Test_tag_line_toolong()
call assert_report(v:exception) call assert_report(v:exception)
catch /.*/ catch /.*/
endtry endtry
call assert_equal('Ignoring long line in tags file', split(execute('messages'), '\n')[-1]) call assert_equal('Searching tags file Xtags', split(execute('messages'), '\n')[-1])
call writefile([ call writefile([
\ '123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567 django/contrib/admin/templates/admin/edit_inline/stacked.html 16;" j line:16 language:HTML' \ '123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567 django/contrib/admin/templates/admin/edit_inline/stacked.html 16;" j line:16 language:HTML'
\ ], 'Xtags') \ ], 'Xtags')
@@ -470,8 +471,26 @@ func Test_tag_line_toolong()
call assert_report(v:exception) call assert_report(v:exception)
catch /.*/ catch /.*/
endtry endtry
call assert_equal('Ignoring long line in tags file', split(execute('messages'), '\n')[-1]) call assert_equal('Searching tags file Xtags', split(execute('messages'), '\n')[-1])
" binary search works in file with long line
call writefile([
\ 'asdfasfd nowhere 16',
\ 'foobar Xsomewhere 3; " 12345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567',
\ 'zasdfasfd nowhere 16',
\ ], 'Xtags')
call writefile([
\ 'one',
\ 'two',
\ 'trhee',
\ 'four',
\ ], 'Xsomewhere')
tag foobar
call assert_equal('Xsomewhere', expand('%'))
call assert_equal(3, getcurpos()[1])
call delete('Xtags') call delete('Xtags')
call delete('Xsomewhere')
set tags& set tags&
let &verbose = old_vbs let &verbose = old_vbs
endfunc endfunc

View File

@@ -741,6 +741,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 */
/**/
2269,
/**/ /**/
2268, 2268,
/**/ /**/