patch 8.1.0544: setting 'filetype' in a modeline causes an error
Problem:    Setting 'filetype' in a modeline causes an error (Hirohito
            Higashi).
Solution:   Don't add the P_INSECURE flag when setting 'filetype' from a
            modeline.  Also for 'syntax'.
			
			
This commit is contained in:
		
							
								
								
									
										84
									
								
								src/option.c
									
									
									
									
									
								
							
							
						
						
									
										84
									
								
								src/option.c
									
									
									
									
									
								
							| @ -3284,7 +3284,7 @@ static char *(p_scl_values[]) = {"yes", "no", "auto", NULL}; | |||||||
| static void set_options_default(int opt_flags); | static void set_options_default(int opt_flags); | ||||||
| static void set_string_default_esc(char *name, char_u *val, int escape); | static void set_string_default_esc(char *name, char_u *val, int escape); | ||||||
| static char_u *term_bg_default(void); | static char_u *term_bg_default(void); | ||||||
| static void did_set_option(int opt_idx, int opt_flags, int new_value); | static void did_set_option(int opt_idx, int opt_flags, int new_value, int value_checked); | ||||||
| static char_u *option_expand(int opt_idx, char_u *val); | static char_u *option_expand(int opt_idx, char_u *val); | ||||||
| static void didset_options(void); | static void didset_options(void); | ||||||
| static void didset_options2(void); | static void didset_options2(void); | ||||||
| @ -3295,7 +3295,7 @@ static long_u *insecure_flag(int opt_idx, int opt_flags); | |||||||
| # define insecure_flag(opt_idx, opt_flags) (&options[opt_idx].flags) | # define insecure_flag(opt_idx, opt_flags) (&options[opt_idx].flags) | ||||||
| #endif | #endif | ||||||
| static void set_string_option_global(int opt_idx, char_u **varp); | static void set_string_option_global(int opt_idx, char_u **varp); | ||||||
| static char_u *did_set_string_option(int opt_idx, char_u **varp, int new_value_alloced, char_u *oldval, char_u *errbuf, int opt_flags); | static char_u *did_set_string_option(int opt_idx, char_u **varp, int new_value_alloced, char_u *oldval, char_u *errbuf, int opt_flags, int *value_checked); | ||||||
| static char_u *set_chars_option(char_u **varp); | static char_u *set_chars_option(char_u **varp); | ||||||
| #ifdef FEAT_CLIPBOARD | #ifdef FEAT_CLIPBOARD | ||||||
| static char_u *check_clipboard_option(void); | static char_u *check_clipboard_option(void); | ||||||
| @ -4706,6 +4706,7 @@ do_set( | |||||||
| 	    else | 	    else | ||||||
| 	    { | 	    { | ||||||
| 		int value_is_replaced = !prepending && !adding && !removing; | 		int value_is_replaced = !prepending && !adding && !removing; | ||||||
|  | 		int value_checked = FALSE; | ||||||
|  |  | ||||||
| 		if (flags & P_BOOL)		    /* boolean */ | 		if (flags & P_BOOL)		    /* boolean */ | ||||||
| 		{ | 		{ | ||||||
| @ -5236,7 +5237,8 @@ do_set( | |||||||
| 			    // or 'filetype' autocommands may be triggered that can | 			    // or 'filetype' autocommands may be triggered that can | ||||||
| 			    // cause havoc. | 			    // cause havoc. | ||||||
| 			    errmsg = did_set_string_option(opt_idx, (char_u **)varp, | 			    errmsg = did_set_string_option(opt_idx, (char_u **)varp, | ||||||
| 				    new_value_alloced, oldval, errbuf, opt_flags); | 				    new_value_alloced, oldval, errbuf, | ||||||
|  | 				    opt_flags, &value_checked); | ||||||
|  |  | ||||||
| 			    if (did_inc_secure) | 			    if (did_inc_secure) | ||||||
| 				--secure; | 				--secure; | ||||||
| @ -5280,7 +5282,8 @@ do_set( | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (opt_idx >= 0) | 		if (opt_idx >= 0) | ||||||
| 		    did_set_option(opt_idx, opt_flags, value_is_replaced); | 		    did_set_option( | ||||||
|  | 			 opt_idx, opt_flags, value_is_replaced, value_checked); | ||||||
| 	    } | 	    } | ||||||
|  |  | ||||||
| skip: | skip: | ||||||
| @ -5348,8 +5351,10 @@ theend: | |||||||
|     static void |     static void | ||||||
| did_set_option( | did_set_option( | ||||||
|     int	    opt_idx, |     int	    opt_idx, | ||||||
|     int	    opt_flags,	    /* possibly with OPT_MODELINE */ |     int	    opt_flags,	    // possibly with OPT_MODELINE | ||||||
|     int	    new_value)	    /* value was replaced completely */ |     int	    new_value,	    // value was replaced completely | ||||||
|  |     int	    value_checked)  // value was checked to be safe, no need to set the | ||||||
|  | 			    // P_INSECURE flag. | ||||||
| { | { | ||||||
|     long_u	*p; |     long_u	*p; | ||||||
|  |  | ||||||
| @ -5359,11 +5364,11 @@ did_set_option( | |||||||
|      * set the P_INSECURE flag.  Otherwise, if a new value is stored reset the |      * set the P_INSECURE flag.  Otherwise, if a new value is stored reset the | ||||||
|      * flag. */ |      * flag. */ | ||||||
|     p = insecure_flag(opt_idx, opt_flags); |     p = insecure_flag(opt_idx, opt_flags); | ||||||
|     if (secure |     if (!value_checked && (secure | ||||||
| #ifdef HAVE_SANDBOX | #ifdef HAVE_SANDBOX | ||||||
| 	    || sandbox != 0 | 	    || sandbox != 0 | ||||||
| #endif | #endif | ||||||
| 	    || (opt_flags & OPT_MODELINE)) | 	    || (opt_flags & OPT_MODELINE))) | ||||||
| 	*p = *p | P_INSECURE; | 	*p = *p | P_INSECURE; | ||||||
|     else if (new_value) |     else if (new_value) | ||||||
| 	*p = *p & ~P_INSECURE; | 	*p = *p & ~P_INSECURE; | ||||||
| @ -6036,6 +6041,7 @@ set_string_option( | |||||||
|     char_u	*saved_newval = NULL; |     char_u	*saved_newval = NULL; | ||||||
| #endif | #endif | ||||||
|     char_u	*r = NULL; |     char_u	*r = NULL; | ||||||
|  |     int		value_checked = FALSE; | ||||||
|  |  | ||||||
|     if (options[opt_idx].var == NULL)	/* don't set hidden option */ |     if (options[opt_idx].var == NULL)	/* don't set hidden option */ | ||||||
| 	return NULL; | 	return NULL; | ||||||
| @ -6063,8 +6069,8 @@ set_string_option( | |||||||
| 	} | 	} | ||||||
| #endif | #endif | ||||||
| 	if ((r = did_set_string_option(opt_idx, varp, TRUE, oldval, NULL, | 	if ((r = did_set_string_option(opt_idx, varp, TRUE, oldval, NULL, | ||||||
| 							   opt_flags)) == NULL) | 					   opt_flags, &value_checked)) == NULL) | ||||||
| 	    did_set_option(opt_idx, opt_flags, TRUE); | 	    did_set_option(opt_idx, opt_flags, TRUE, value_checked); | ||||||
|  |  | ||||||
| #if defined(FEAT_EVAL) | #if defined(FEAT_EVAL) | ||||||
| 	/* call autocommand after handling side effects */ | 	/* call autocommand after handling side effects */ | ||||||
| @ -6099,12 +6105,14 @@ valid_filetype(char_u *val) | |||||||
|  */ |  */ | ||||||
|     static char_u * |     static char_u * | ||||||
| did_set_string_option( | did_set_string_option( | ||||||
|     int		opt_idx,		/* index in options[] table */ |     int		opt_idx,		// index in options[] table | ||||||
|     char_u	**varp,			/* pointer to the option variable */ |     char_u	**varp,			// pointer to the option variable | ||||||
|     int		new_value_alloced,	/* new value was allocated */ |     int		new_value_alloced,	// new value was allocated | ||||||
|     char_u	*oldval,		/* previous value of the option */ |     char_u	*oldval,		// previous value of the option | ||||||
|     char_u	*errbuf,		/* buffer for errors, or NULL */ |     char_u	*errbuf,		// buffer for errors, or NULL | ||||||
|     int		opt_flags)		/* OPT_LOCAL and/or OPT_GLOBAL */ |     int		opt_flags,		// OPT_LOCAL and/or OPT_GLOBAL | ||||||
|  |     int		*value_checked)		// value was checked to be save, no | ||||||
|  | 					// need to set P_INSECURE | ||||||
| { | { | ||||||
|     char_u	*errmsg = NULL; |     char_u	*errmsg = NULL; | ||||||
|     char_u	*s, *p; |     char_u	*s, *p; | ||||||
| @ -6134,10 +6142,9 @@ did_set_string_option( | |||||||
| 	errmsg = e_secure; | 	errmsg = e_secure; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /* Check for a "normal" directory or file name in some options.  Disallow a |     // Check for a "normal" directory or file name in some options.  Disallow a | ||||||
|      * path separator (slash and/or backslash), wildcards and characters that |     // path separator (slash and/or backslash), wildcards and characters that | ||||||
|      * are often illegal in a file name. Be more permissive if "secure" is off. |     // are often illegal in a file name. Be more permissive if "secure" is off. | ||||||
|      */ |  | ||||||
|     else if (((options[opt_idx].flags & P_NFNAME) |     else if (((options[opt_idx].flags & P_NFNAME) | ||||||
| 		    && vim_strpbrk(*varp, (char_u *)(secure | 		    && vim_strpbrk(*varp, (char_u *)(secure | ||||||
| 			    ? "/\\*?[|;&<>\r\n" : "/\\*?[<>\r\n")) != NULL) | 			    ? "/\\*?[|;&<>\r\n" : "/\\*?[<>\r\n")) != NULL) | ||||||
| @ -6524,9 +6531,23 @@ did_set_string_option( | |||||||
| 	if (!valid_filetype(*varp)) | 	if (!valid_filetype(*varp)) | ||||||
| 	    errmsg = e_invarg; | 	    errmsg = e_invarg; | ||||||
| 	else | 	else | ||||||
| 	    /* load or unload key mapping tables */ | 	{ | ||||||
|  | 	    int	    secure_save = secure; | ||||||
|  |  | ||||||
|  | 	    // Reset the secure flag, since the value of 'keymap' has | ||||||
|  | 	    // been checked to be safe. | ||||||
|  | 	    secure = 0; | ||||||
|  |  | ||||||
|  | 	    // load or unload key mapping tables | ||||||
| 	    errmsg = keymap_init(); | 	    errmsg = keymap_init(); | ||||||
|  |  | ||||||
|  | 	    secure = secure_save; | ||||||
|  |  | ||||||
|  | 	    // Since we check the value, there is no need to set P_INSECURE, | ||||||
|  | 	    // even when the value comes from a modeline. | ||||||
|  | 	    *value_checked = TRUE; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if (errmsg == NULL) | 	if (errmsg == NULL) | ||||||
| 	{ | 	{ | ||||||
| 	    if (*curbuf->b_p_keymap != NUL) | 	    if (*curbuf->b_p_keymap != NUL) | ||||||
| @ -7523,7 +7544,13 @@ did_set_string_option( | |||||||
| 	if (!valid_filetype(*varp)) | 	if (!valid_filetype(*varp)) | ||||||
| 	    errmsg = e_invarg; | 	    errmsg = e_invarg; | ||||||
| 	else | 	else | ||||||
|  | 	{ | ||||||
| 	    value_changed = STRCMP(oldval, *varp) != 0; | 	    value_changed = STRCMP(oldval, *varp) != 0; | ||||||
|  |  | ||||||
|  | 	    // Since we check the value, there is no need to set P_INSECURE, | ||||||
|  | 	    // even when the value comes from a modeline. | ||||||
|  | 	    *value_checked = TRUE; | ||||||
|  | 	} | ||||||
|     } |     } | ||||||
|  |  | ||||||
| #ifdef FEAT_SYN_HL | #ifdef FEAT_SYN_HL | ||||||
| @ -7532,7 +7559,13 @@ did_set_string_option( | |||||||
| 	if (!valid_filetype(*varp)) | 	if (!valid_filetype(*varp)) | ||||||
| 	    errmsg = e_invarg; | 	    errmsg = e_invarg; | ||||||
| 	else | 	else | ||||||
|  | 	{ | ||||||
| 	    value_changed = STRCMP(oldval, *varp) != 0; | 	    value_changed = STRCMP(oldval, *varp) != 0; | ||||||
|  |  | ||||||
|  | 	    // Since we check the value, there is no need to set P_INSECURE, | ||||||
|  | 	    // even when the value comes from a modeline. | ||||||
|  | 	    *value_checked = TRUE; | ||||||
|  | 	} | ||||||
|     } |     } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| @ -7752,7 +7785,12 @@ did_set_string_option( | |||||||
| 	     * already set to this value. */ | 	     * already set to this value. */ | ||||||
| 	    if (!(opt_flags & OPT_MODELINE) || value_changed) | 	    if (!(opt_flags & OPT_MODELINE) || value_changed) | ||||||
| 	    { | 	    { | ||||||
| 		static int ft_recursive = 0; | 		static int  ft_recursive = 0; | ||||||
|  | 		int	    secure_save = secure; | ||||||
|  |  | ||||||
|  | 		// Reset the secure flag, since the value of 'filetype' has | ||||||
|  | 		// been checked to be safe. | ||||||
|  | 		secure = 0; | ||||||
|  |  | ||||||
| 		++ft_recursive; | 		++ft_recursive; | ||||||
| 		did_filetype = TRUE; | 		did_filetype = TRUE; | ||||||
| @ -7764,6 +7802,8 @@ did_set_string_option( | |||||||
| 		/* Just in case the old "curbuf" is now invalid. */ | 		/* Just in case the old "curbuf" is now invalid. */ | ||||||
| 		if (varp != &(curbuf->b_p_ft)) | 		if (varp != &(curbuf->b_p_ft)) | ||||||
| 		    varp = NULL; | 		    varp = NULL; | ||||||
|  |  | ||||||
|  | 		secure = secure_save; | ||||||
| 	    } | 	    } | ||||||
| 	} | 	} | ||||||
| #ifdef FEAT_SPELL | #ifdef FEAT_SPELL | ||||||
|  | |||||||
| @ -6,7 +6,83 @@ func Test_modeline_invalid() | |||||||
|   let modeline = &modeline |   let modeline = &modeline | ||||||
|   set modeline |   set modeline | ||||||
|   call assert_fails('split Xmodeline', 'E518:') |   call assert_fails('split Xmodeline', 'E518:') | ||||||
|  |  | ||||||
|   let &modeline = modeline |   let &modeline = modeline | ||||||
|   bwipe! |   bwipe! | ||||||
|   call delete('Xmodeline') |   call delete('Xmodeline') | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_filetype() | ||||||
|  |   call writefile(['vim: set ft=c :', 'nothing'], 'Xmodeline_filetype') | ||||||
|  |   let modeline = &modeline | ||||||
|  |   set modeline | ||||||
|  |   filetype plugin on | ||||||
|  |   split Xmodeline_filetype | ||||||
|  |   call assert_equal("c", &filetype) | ||||||
|  |   call assert_equal(1, b:did_ftplugin) | ||||||
|  |   call assert_equal("ccomplete#Complete", &ofu) | ||||||
|  |  | ||||||
|  |   bwipe! | ||||||
|  |   call delete('Xmodeline_filetype') | ||||||
|  |   let &modeline = modeline | ||||||
|  |   filetype plugin off | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_syntax() | ||||||
|  |   call writefile(['vim: set syn=c :', 'nothing'], 'Xmodeline_syntax') | ||||||
|  |   let modeline = &modeline | ||||||
|  |   set modeline | ||||||
|  |   syntax enable | ||||||
|  |   split Xmodeline_syntax | ||||||
|  |   call assert_equal("c", &syntax) | ||||||
|  |   call assert_equal("c", b:current_syntax) | ||||||
|  |  | ||||||
|  |   bwipe! | ||||||
|  |   call delete('Xmodeline_syntax') | ||||||
|  |   let &modeline = modeline | ||||||
|  |   syntax off | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_keymap() | ||||||
|  |   call writefile(['vim: set keymap=greek :', 'nothing'], 'Xmodeline_keymap') | ||||||
|  |   let modeline = &modeline | ||||||
|  |   set modeline | ||||||
|  |   split Xmodeline_keymap | ||||||
|  |   call assert_equal("greek", &keymap) | ||||||
|  |   call assert_match('greek\|grk', b:keymap_name) | ||||||
|  |  | ||||||
|  |   bwipe! | ||||||
|  |   call delete('Xmodeline_keymap') | ||||||
|  |   let &modeline = modeline | ||||||
|  |   set keymap= iminsert=0 imsearch=-1 | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func s:modeline_fails(what, text) | ||||||
|  |   let fname = "Xmodeline_fails_" . a:what | ||||||
|  |   call writefile(['vim: set ' . a:text . ' :', 'nothing'], fname) | ||||||
|  |   let modeline = &modeline | ||||||
|  |   set modeline | ||||||
|  |   filetype plugin on | ||||||
|  |   syntax enable | ||||||
|  |   call assert_fails('split ' . fname, 'E474:') | ||||||
|  |   call assert_equal("", &filetype) | ||||||
|  |   call assert_equal("", &syntax) | ||||||
|  |  | ||||||
|  |   bwipe! | ||||||
|  |   call delete(fname) | ||||||
|  |   let &modeline = modeline | ||||||
|  |   filetype plugin off | ||||||
|  |   syntax off | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_filetype_fails() | ||||||
|  |   call s:modeline_fails('filetype', 'ft=evil$CMD') | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_syntax_fails() | ||||||
|  |   call s:modeline_fails('syntax', 'syn=evil$CMD') | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | func Test_modeline_keymap_fails() | ||||||
|  |   call s:modeline_fails('keymap', 'keymap=evil$CMD') | ||||||
|  | endfunc | ||||||
|  | |||||||
| @ -792,6 +792,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 */ | ||||||
|  | /**/ | ||||||
|  |     544, | ||||||
| /**/ | /**/ | ||||||
|     543, |     543, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user