patch 9.1.1077: included syntax items do not understand contains=TOP

Problem:  Syntax engine interpreted contains=TOP as matching nothing
          inside included files, since :syn-include forces HL_CONTAINED
          on for every included item. After 8.2.2761, interprets
          contains=TOP as contains=@INCLUDED, which is also not correct
          since it doesn't respect exclusions, and doesn't work if there
          is no @INCLUDED cluster.
Solution: revert patch 8.2.2761, instead track groups that have had
          HL_CONTAINED forced, and interpret contains=TOP and
          contains=CONTAINED using this. (Theodore Dubois)

fixes: #11277
closes: #16571

Signed-off-by: Theodore Dubois <tblodt@icloud.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
This commit is contained in:
Theodore Dubois
2025-02-05 23:59:25 +01:00
committed by Christian Brabandt
parent 34e1e8de91
commit f50d5364d7
4 changed files with 38 additions and 20 deletions

View File

@ -299,7 +299,7 @@ static void update_si_attr(int idx);
static void check_keepend(void); static void check_keepend(void);
static void update_si_end(stateitem_T *sip, int startcol, int force); static void update_si_end(stateitem_T *sip, int startcol, int force);
static short *copy_id_list(short *list); static short *copy_id_list(short *list);
static int in_id_list(stateitem_T *item, short *cont_list, struct sp_syn *ssp, int contained); static int in_id_list(stateitem_T *item, short *cont_list, struct sp_syn *ssp, int flags);
static int push_current_state(int idx); static int push_current_state(int idx);
static void pop_current_state(void); static void pop_current_state(void);
#ifdef FEAT_PROFILE #ifdef FEAT_PROFILE
@ -1943,7 +1943,7 @@ syn_current_attr(
? !(spp->sp_flags & HL_CONTAINED) ? !(spp->sp_flags & HL_CONTAINED)
: in_id_list(cur_si, : in_id_list(cur_si,
cur_si->si_cont_list, &spp->sp_syn, cur_si->si_cont_list, &spp->sp_syn,
spp->sp_flags & HL_CONTAINED)))) spp->sp_flags))))
{ {
int r; int r;
@ -3269,7 +3269,7 @@ check_keyword_id(
: (cur_si == NULL : (cur_si == NULL
? !(kp->flags & HL_CONTAINED) ? !(kp->flags & HL_CONTAINED)
: in_id_list(cur_si, cur_si->si_cont_list, : in_id_list(cur_si, cur_si->si_cont_list,
&kp->k_syn, kp->flags & HL_CONTAINED))) &kp->k_syn, kp->flags)))
{ {
*endcolp = startcol + kwlen; *endcolp = startcol + kwlen;
*flagsp = kp->flags; *flagsp = kp->flags;
@ -4681,7 +4681,7 @@ syn_incl_toplevel(int id, int *flagsp)
{ {
if ((*flagsp & HL_CONTAINED) || curwin->w_s->b_syn_topgrp == 0) if ((*flagsp & HL_CONTAINED) || curwin->w_s->b_syn_topgrp == 0)
return; return;
*flagsp |= HL_CONTAINED; *flagsp |= HL_CONTAINED | HL_INCLUDED_TOPLEVEL;
if (curwin->w_s->b_syn_topgrp >= SYNID_CLUSTER) if (curwin->w_s->b_syn_topgrp >= SYNID_CLUSTER)
{ {
// We have to alloc this, because syn_combine_list() will free it. // We have to alloc this, because syn_combine_list() will free it.
@ -5969,17 +5969,12 @@ get_id_list(
break; break;
} }
if (name[1] == 'A') if (name[1] == 'A')
id = SYNID_ALLBUT + current_syn_inc_tag; id = SYNID_ALLBUT;
else if (name[1] == 'T') else if (name[1] == 'T')
{ id = SYNID_TOP;
if (curwin->w_s->b_syn_topgrp >= SYNID_CLUSTER)
id = curwin->w_s->b_syn_topgrp;
else
id = SYNID_TOP + current_syn_inc_tag;
}
else else
id = SYNID_CONTAINED + current_syn_inc_tag; id = SYNID_CONTAINED;
id += current_syn_inc_tag;
} }
else if (name[1] == '@') else if (name[1] == '@')
{ {
@ -6127,7 +6122,7 @@ in_id_list(
stateitem_T *cur_si, // current item or NULL stateitem_T *cur_si, // current item or NULL
short *list, // id list short *list, // id list
struct sp_syn *ssp, // group id and ":syn include" tag of group struct sp_syn *ssp, // group id and ":syn include" tag of group
int contained) // group id is contained int flags) // group flags
{ {
int retval; int retval;
short *scl_list; short *scl_list;
@ -6135,6 +6130,7 @@ in_id_list(
short id = ssp->id; short id = ssp->id;
static int depth = 0; static int depth = 0;
int r; int r;
int toplevel;
// If ssp has a "containedin" list and "cur_si" is in it, return TRUE. // If ssp has a "containedin" list and "cur_si" is in it, return TRUE.
if (cur_si != NULL && ssp->cont_in_list != NULL if (cur_si != NULL && ssp->cont_in_list != NULL
@ -6148,7 +6144,7 @@ in_id_list(
// cur_si->si_idx is -1 for keywords, these never contain anything. // cur_si->si_idx is -1 for keywords, these never contain anything.
if (cur_si->si_idx >= 0 && in_id_list(NULL, ssp->cont_in_list, if (cur_si->si_idx >= 0 && in_id_list(NULL, ssp->cont_in_list,
&(SYN_ITEMS(syn_block)[cur_si->si_idx].sp_syn), &(SYN_ITEMS(syn_block)[cur_si->si_idx].sp_syn),
SYN_ITEMS(syn_block)[cur_si->si_idx].sp_flags & HL_CONTAINED)) SYN_ITEMS(syn_block)[cur_si->si_idx].sp_flags))
return TRUE; return TRUE;
} }
@ -6160,7 +6156,14 @@ in_id_list(
* inside anything. Only allow not-contained groups. * inside anything. Only allow not-contained groups.
*/ */
if (list == ID_LIST_ALL) if (list == ID_LIST_ALL)
return !contained; return !(flags & HL_CONTAINED);
/*
* Is this top-level (i.e. not 'contained') in the file it was declared in?
* For included files, this is different from HL_CONTAINED, which is set
* unconditionally.
*/
toplevel = !(flags & HL_CONTAINED) || (flags & HL_INCLUDED_TOPLEVEL);
/* /*
* If the first item is "ALLBUT", return TRUE if "id" is NOT in the * If the first item is "ALLBUT", return TRUE if "id" is NOT in the
@ -6179,13 +6182,13 @@ in_id_list(
else if (item < SYNID_CONTAINED) else if (item < SYNID_CONTAINED)
{ {
// TOP: accept all not-contained groups in the same file // TOP: accept all not-contained groups in the same file
if (item - SYNID_TOP != ssp->inc_tag || contained) if (item - SYNID_TOP != ssp->inc_tag || !toplevel)
return FALSE; return FALSE;
} }
else else
{ {
// CONTAINED: accept all contained groups in the same file // CONTAINED: accept all contained groups in the same file
if (item - SYNID_CONTAINED != ssp->inc_tag || !contained) if (item - SYNID_CONTAINED != ssp->inc_tag || toplevel)
return FALSE; return FALSE;
} }
item = *++list; item = *++list;
@ -6209,7 +6212,7 @@ in_id_list(
if (scl_list != NULL && depth < 30) if (scl_list != NULL && depth < 30)
{ {
++depth; ++depth;
r = in_id_list(NULL, scl_list, ssp, contained); r = in_id_list(NULL, scl_list, ssp, flags);
--depth; --depth;
if (r) if (r)
return retval; return retval;

View File

@ -949,7 +949,7 @@ func Test_syn_contained_transparent()
endfunc endfunc
func Test_syn_include_contains_TOP() func Test_syn_include_contains_TOP()
let l:case = "TOP in included syntax means its group list name" let l:case = "TOP in included syntax refers to top level of that included syntax"
new new
syntax include @INCLUDED syntax/c.vim syntax include @INCLUDED syntax/c.vim
syntax region FencedCodeBlockC start=/```c/ end=/```/ contains=@INCLUDED syntax region FencedCodeBlockC start=/```c/ end=/```/ contains=@INCLUDED
@ -964,6 +964,18 @@ func Test_syn_include_contains_TOP()
bw! bw!
endfunc endfunc
func Test_syn_include_contains_TOP_excluding()
new
syntax include @INCLUDED syntax/c.vim
syntax region FencedCodeBlockC start=/```c/ end=/```/ contains=@INCLUDED
call setline(1, ['```c', '#if 0', 'int', '#else', 'int', '#if', '#endif', '```' ])
let l:expected = ["cCppOutElse", "cConditional"]
eval AssertHighlightGroups(6, 1, l:expected, 1)
syntax clear
bw!
endfunc
" This was using freed memory " This was using freed memory
func Test_WinEnter_synstack_synID() func Test_WinEnter_synstack_synID()
autocmd WinEnter * call synstack(line("."), col(".")) autocmd WinEnter * call synstack(line("."), col("."))

View File

@ -704,6 +704,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 */
/**/
1077,
/**/ /**/
1076, 1076,
/**/ /**/

View File

@ -953,6 +953,7 @@ extern int (*dyn_libintl_wputenv)(const wchar_t *envstring);
# define HL_TRANS_CONT 0x10000 // transparent item without contains arg # define HL_TRANS_CONT 0x10000 // transparent item without contains arg
# define HL_CONCEAL 0x20000 // can be concealed # define HL_CONCEAL 0x20000 // can be concealed
# define HL_CONCEALENDS 0x40000 // can be concealed # define HL_CONCEALENDS 0x40000 // can be concealed
# define HL_INCLUDED_TOPLEVEL 0x80000 // toplevel item in included syntax, allowed by contains=TOP
#endif #endif
// Values for 'options' argument in do_search() and searchit() // Values for 'options' argument in do_search() and searchit()