vim-patch:8.1.1366: using expressions in a modeline is unsafe

Problem:    Using expressions in a modeline is unsafe.
Solution:   Disallow using expressions in a modeline, unless the
            'modelineexpr' option is set.  Update help, add more tests.
110289e781
This commit is contained in:
James McCoy 2019-06-22 21:17:53 -04:00
parent 45bb1757bf
commit 1e4673d167
No known key found for this signature in database
GPG Key ID: DFE691AE331BA3DB
7 changed files with 176 additions and 27 deletions

View File

@ -478,14 +478,17 @@ backslash in front of the ':' will be removed. Example:
/* vi:set dir=c\:\tmp: */ ~ /* vi:set dir=c\:\tmp: */ ~
This sets the 'dir' option to "c:\tmp". Only a single backslash before the This sets the 'dir' option to "c:\tmp". Only a single backslash before the
':' is removed. Thus to include "\:" you have to specify "\\:". ':' is removed. Thus to include "\:" you have to specify "\\:".
*E992*
No other commands than "set" are supported, for security reasons (somebody No other commands than "set" are supported, for security reasons (somebody
might create a Trojan horse text file with modelines). And not all options might create a Trojan horse text file with modelines). And not all options
can be set. For some options a flag is set, so that when it's used the can be set. For some options a flag is set, so that when the value is used
|sandbox| is effective. Still, there is always a small risk that a modeline the |sandbox| is effective. Some options can only be set from the modeline
causes trouble. E.g., when some joker sets 'textwidth' to 5 all your lines when 'modelineexpr' is set (the default is off).
are wrapped unexpectedly. So disable modelines before editing untrusted text.
The mail ftplugin does this, for example. Still, there is always a small risk that a modeline causes trouble. E.g.,
when some joker sets 'textwidth' to 5 all your lines are wrapped unexpectedly.
So disable modelines before editing untrusted text. The mail ftplugin does
this, for example.
Hint: If you would like to do something else than setting an option, you could Hint: If you would like to do something else than setting an option, you could
define an autocommand that checks the file for a specific string. For define an autocommand that checks the file for a specific string. For
@ -2427,7 +2430,7 @@ A jump table for the options with a short description can be found at |Q_op|.
The expression will be evaluated in the |sandbox| if set from a The expression will be evaluated in the |sandbox| if set from a
modeline, see |sandbox-option|. modeline, see |sandbox-option|.
This option can't be set from a |modeline| when the 'diff' option is This option can't be set from a |modeline| when the 'diff' option is
on. on or the 'modelineexpr' option is off.
It is not allowed to change text or jump to another window while It is not allowed to change text or jump to another window while
evaluating 'foldexpr' |textlock|. evaluating 'foldexpr' |textlock|.
@ -2542,6 +2545,7 @@ A jump table for the options with a short description can be found at |Q_op|.
The expression will be evaluated in the |sandbox| if set from a The expression will be evaluated in the |sandbox| if set from a
modeline, see |sandbox-option|. modeline, see |sandbox-option|.
This option cannot be set in a modeline when 'modelineexpr' is off.
It is not allowed to change text or jump to another window while It is not allowed to change text or jump to another window while
evaluating 'foldtext' |textlock|. evaluating 'foldtext' |textlock|.
@ -2577,16 +2581,8 @@ A jump table for the options with a short description can be found at |Q_op|.
The expression will be evaluated in the |sandbox| when set from a The expression will be evaluated in the |sandbox| when set from a
modeline, see |sandbox-option|. That stops the option from working, modeline, see |sandbox-option|. That stops the option from working,
since changing the buffer text is not allowed. since changing the buffer text is not allowed.
This option cannot be set in a modeline when 'modelineexpr' is off.
*'formatoptions'* *'fo'* NOTE: This option is set to "" when 'compatible' is set.
'formatoptions' 'fo' string (default: "tcqj", Vi default: "vt")
local to buffer
This is a sequence of letters which describes how automatic
formatting is to be done. See |fo-table|. When the 'paste' option is
on, no formatting is done (like 'formatoptions' is empty). Commas can
be inserted for readability.
To avoid problems with flags that are added in the future, use the
"+=" and "-=" feature of ":set" |add-option-flags|.
*'formatlistpat'* *'flp'* *'formatlistpat'* *'flp'*
'formatlistpat' 'flp' string (default: "^\s*\d\+[\]:.)}\t ]\s*") 'formatlistpat' 'flp' string (default: "^\s*\d\+[\]:.)}\t ]\s*")
@ -2601,6 +2597,16 @@ A jump table for the options with a short description can be found at |Q_op|.
The default recognizes a number, followed by an optional punctuation The default recognizes a number, followed by an optional punctuation
character and white space. character and white space.
*'formatoptions'* *'fo'*
'formatoptions' 'fo' string (default: "tcqj", Vi default: "vt")
local to buffer
This is a sequence of letters which describes how automatic
formatting is to be done. See |fo-table|. When the 'paste' option is
on, no formatting is done (like 'formatoptions' is empty). Commas can
be inserted for readability.
To avoid problems with flags that are added in the future, use the
"+=" and "-=" feature of ":set" |add-option-flags|.
*'formatprg'* *'fp'* *'formatprg'* *'fp'*
'formatprg' 'fp' string (default "") 'formatprg' 'fp' string (default "")
global or local to buffer |global-local| global or local to buffer |global-local|
@ -2631,6 +2637,9 @@ A jump table for the options with a short description can be found at |Q_op|.
- system signals low battery life - system signals low battery life
- Nvim exits abnormally - Nvim exits abnormally
This option cannot be set from a |modeline| or in the |sandbox|, for
security reasons.
*'gdefault'* *'gd'* *'nogdefault'* *'nogd'* *'gdefault'* *'gd'* *'nogdefault'* *'nogd'*
'gdefault' 'gd' boolean (default off) 'gdefault' 'gd' boolean (default off)
global global
@ -2972,6 +2981,7 @@ A jump table for the options with a short description can be found at |Q_op|.
'guitabtooltip' is used for the tooltip, see below. 'guitabtooltip' is used for the tooltip, see below.
The expression will be evaluated in the |sandbox| when set from a The expression will be evaluated in the |sandbox| when set from a
modeline, see |sandbox-option|. modeline, see |sandbox-option|.
This option cannot be set in a modeline when 'modelineexpr' is off.
Only used when the GUI tab pages line is displayed. 'e' must be Only used when the GUI tab pages line is displayed. 'e' must be
present in 'guioptions'. For the non-GUI tab pages line 'tabline' is present in 'guioptions'. For the non-GUI tab pages line 'tabline' is
@ -3100,6 +3110,7 @@ A jump table for the options with a short description can be found at |Q_op|.
When this option contains printf-style '%' items, they will be When this option contains printf-style '%' items, they will be
expanded according to the rules used for 'statusline'. See expanded according to the rules used for 'statusline'. See
'titlestring' for example settings. 'titlestring' for example settings.
This option cannot be set in a modeline when 'modelineexpr' is off.
*'ignorecase'* *'ic'* *'noignorecase'* *'noic'* *'ignorecase'* *'ic'* *'noignorecase'* *'noic'*
'ignorecase' 'ic' boolean (default off) 'ignorecase' 'ic' boolean (default off)
@ -3203,6 +3214,7 @@ A jump table for the options with a short description can be found at |Q_op|.
The expression will be evaluated in the |sandbox| when set from a The expression will be evaluated in the |sandbox| when set from a
modeline, see |sandbox-option|. modeline, see |sandbox-option|.
This option cannot be set in a modeline when 'modelineexpr' is off.
It is not allowed to change text or jump to another window while It is not allowed to change text or jump to another window while
evaluating 'includeexpr' |textlock|. evaluating 'includeexpr' |textlock|.
@ -3271,6 +3283,7 @@ A jump table for the options with a short description can be found at |Q_op|.
The expression will be evaluated in the |sandbox| when set from a The expression will be evaluated in the |sandbox| when set from a
modeline, see |sandbox-option|. modeline, see |sandbox-option|.
This option cannot be set in a modeline when 'modelineexpr' is off.
It is not allowed to change text or jump to another window while It is not allowed to change text or jump to another window while
evaluating 'indentexpr' |textlock|. evaluating 'indentexpr' |textlock|.
@ -3777,6 +3790,12 @@ A jump table for the options with a short description can be found at |Q_op|.
< This option cannot be set from a |modeline| or in the |sandbox|, for < This option cannot be set from a |modeline| or in the |sandbox|, for
security reasons. security reasons.
*'makespellmem'* *'msm'*
'makespellmem' 'msm' string (default "460000,2000,500")
global
Values relevant only when compressing a spell file, see |spell|.
This option cannot be set from a |modeline| or in the |sandbox|.
*'matchpairs'* *'mps'* *'matchpairs'* *'mps'*
'matchpairs' 'mps' string (default "(:),{:},[:]") 'matchpairs' 'mps' string (default "(:),{:},[:]")
local to buffer local to buffer
@ -3892,6 +3911,17 @@ A jump table for the options with a short description can be found at |Q_op|.
'modeline' 'ml' boolean (Vim default: on (off for root), 'modeline' 'ml' boolean (Vim default: on (off for root),
Vi default: off) Vi default: off)
local to buffer local to buffer
If 'modeline' is on 'modelines' gives the number of lines that is
checked for set commands. If 'modeline' is off or 'modelines' is zero
no lines are checked. See |modeline|.
*'modelineexpr'* *'mle'* *'nomodelineexpr'* *'nomle'*
'modelineexpr' 'mle' boolean (default: off)
global
When on allow some options that are an expression to be set in the
modeline. Check the option for whether it is affected by
'modelineexpr'. Also see |modeline|.
*'modelines'* *'mls'* *'modelines'* *'mls'*
'modelines' 'mls' number (default 5) 'modelines' 'mls' number (default 5)
global global
@ -4668,6 +4698,8 @@ A jump table for the options with a short description can be found at |Q_op|.
When this option is not empty, it determines the content of the ruler When this option is not empty, it determines the content of the ruler
string, as displayed for the 'ruler' option. string, as displayed for the 'ruler' option.
The format of this option is like that of 'statusline'. The format of this option is like that of 'statusline'.
This option cannot be set in a modeline when 'modelineexpr' is off.
The default ruler width is 17 characters. To make the ruler 15 The default ruler width is 17 characters. To make the ruler 15
characters wide, put "%15(" at the start and "%)" at the end. characters wide, put "%15(" at the start and "%)" at the end.
Example: > Example: >
@ -5065,6 +5097,8 @@ A jump table for the options with a short description can be found at |Q_op|.
When equal to "NONE" no shada file will be read or written. When equal to "NONE" no shada file will be read or written.
This option can be set with the |-i| command line flag. The |--clean| This option can be set with the |-i| command line flag. The |--clean|
command line flag sets it to "NONE". command line flag sets it to "NONE".
This option cannot be set from a |modeline| or in the |sandbox|, for
security reasons.
*'shell'* *'sh'* *E91* *'shell'* *'sh'* *E91*
'shell' 'sh' string (default $SHELL or "sh", 'shell' 'sh' string (default $SHELL or "sh",
@ -5304,7 +5338,8 @@ A jump table for the options with a short description can be found at |Q_op|.
"Pattern not found", "Back at original", etc. "Pattern not found", "Back at original", etc.
q use "recording" instead of "recording @a" q use "recording" instead of "recording @a"
F don't give the file info when editing a file, like `:silent` F don't give the file info when editing a file, like `:silent`
was used for the command was used for the command; note that this also affects messages
from autocommands
S do not show search count message when searching, e.g. S do not show search count message when searching, e.g.
"[1/5]" "[1/5]"
@ -5831,6 +5866,7 @@ A jump table for the options with a short description can be found at |Q_op|.
The 'statusline' option will be evaluated in the |sandbox| if set from The 'statusline' option will be evaluated in the |sandbox| if set from
a modeline, see |sandbox-option|. a modeline, see |sandbox-option|.
This option cannot be set in a modeline when 'modelineexpr' is off.
It is not allowed to change text or jump to another window while It is not allowed to change text or jump to another window while
evaluating 'statusline' |textlock|. evaluating 'statusline' |textlock|.
@ -5985,6 +6021,8 @@ A jump table for the options with a short description can be found at |Q_op|.
the text to be displayed. Use "%1T" for the first label, "%2T" for the text to be displayed. Use "%1T" for the first label, "%2T" for
the second one, etc. Use "%X" items for closing labels. the second one, etc. Use "%X" items for closing labels.
This option cannot be set in a modeline when 'modelineexpr' is off.
Keep in mind that only one of the tab pages is the current one, others Keep in mind that only one of the tab pages is the current one, others
are invisible and you can't jump to their windows. are invisible and you can't jump to their windows.
@ -6264,8 +6302,11 @@ A jump table for the options with a short description can be found at |Q_op|.
global global
When this option is not empty, it will be used for the title of the When this option is not empty, it will be used for the title of the
window. This happens only when the 'title' option is on. window. This happens only when the 'title' option is on.
When this option contains printf-style '%' items, they will be When this option contains printf-style '%' items, they will be
expanded according to the rules used for 'statusline'. expanded according to the rules used for 'statusline'.
This option cannot be set in a modeline when 'modelineexpr' is off.
Example: > Example: >
:auto BufEnter * let &titlestring = hostname() . "/" . expand("%:p") :auto BufEnter * let &titlestring = hostname() . "/" . expand("%:p")
:set title titlestring=%<%F%=%l/%L-%P titlelen=70 :set title titlestring=%<%F%=%l/%L-%P titlelen=70
@ -6299,6 +6340,8 @@ A jump table for the options with a short description can be found at |Q_op|.
undo file that exists is used. When it cannot be read an error is undo file that exists is used. When it cannot be read an error is
given, no further entry is used. given, no further entry is used.
See |undo-persistence|. See |undo-persistence|.
This option cannot be set from a |modeline| or in the |sandbox|, for
security reasons.
*'undofile'* *'noundofile'* *'udf'* *'noudf'* *'undofile'* *'noundofile'* *'udf'* *'noudf'*
'undofile' 'udf' boolean (default off) 'undofile' 'udf' boolean (default off)

View File

@ -79,6 +79,7 @@ local get_flags = function(o)
{'pri_mkrc'}, {'pri_mkrc'},
{'deny_in_modelines', 'P_NO_ML'}, {'deny_in_modelines', 'P_NO_ML'},
{'deny_duplicates', 'P_NODUP'}, {'deny_duplicates', 'P_NODUP'},
{'modelineexpr', 'P_MLE'},
}) do }) do
local key_name = flag_desc[1] local key_name = flag_desc[1]
local def_name = flag_desc[2] or ('P_' .. key_name:upper()) local def_name = flag_desc[2] or ('P_' .. key_name:upper())

View File

@ -253,6 +253,7 @@ typedef struct vimoption {
#define P_RWINONLY 0x10000000U ///< only redraw current window #define P_RWINONLY 0x10000000U ///< only redraw current window
#define P_NDNAME 0x20000000U ///< only normal dir name chars allowed #define P_NDNAME 0x20000000U ///< only normal dir name chars allowed
#define P_UI_OPTION 0x40000000U ///< send option to remote ui #define P_UI_OPTION 0x40000000U ///< send option to remote ui
#define P_MLE 0x80000000U ///< under control of 'modelineexpr'
#define HIGHLIGHT_INIT \ #define HIGHLIGHT_INIT \
"8:SpecialKey,~:EndOfBuffer,z:TermCursor,Z:TermCursorNC,@:NonText," \ "8:SpecialKey,~:EndOfBuffer,z:TermCursor,Z:TermCursorNC,@:NonText," \
@ -1327,6 +1328,11 @@ int do_set(
errmsg = (char_u *)_("E520: Not allowed in a modeline"); errmsg = (char_u *)_("E520: Not allowed in a modeline");
goto skip; goto skip;
} }
if ((flags & P_MLE) && !p_mle) {
errmsg = (char_u *)_(
"E992: Not allowed in a modeline when 'modelineexpr' is off");
goto skip;
}
// In diff mode some options are overruled. This avoids that // In diff mode some options are overruled. This avoids that
// 'foldmethod' becomes "marker" instead of "diff" and that // 'foldmethod' becomes "marker" instead of "diff" and that
// "wrap" gets set. // "wrap" gets set.

View File

@ -496,6 +496,7 @@ EXTERN long p_mmd; // 'maxmapdepth'
EXTERN long p_mmp; // 'maxmempattern' EXTERN long p_mmp; // 'maxmempattern'
EXTERN long p_mis; // 'menuitems' EXTERN long p_mis; // 'menuitems'
EXTERN char_u *p_msm; // 'mkspellmem' EXTERN char_u *p_msm; // 'mkspellmem'
EXTERN long p_mle; // 'modelineexpr'
EXTERN long p_mls; // 'modelines' EXTERN long p_mls; // 'modelines'
EXTERN char_u *p_mouse; // 'mouse' EXTERN char_u *p_mouse; // 'mouse'
EXTERN char_u *p_mousem; // 'mousemodel' EXTERN char_u *p_mousem; // 'mousemodel'

View File

@ -8,6 +8,7 @@
-- defaults={condition=nil, if_true={vi=224, vim=0}, if_false=nil}, -- defaults={condition=nil, if_true={vi=224, vim=0}, if_false=nil},
-- secure=nil, gettext=nil, noglob=nil, normal_fname_chars=nil, -- secure=nil, gettext=nil, noglob=nil, normal_fname_chars=nil,
-- pri_mkrc=nil, deny_in_modelines=nil, normal_dname_chars=nil, -- pri_mkrc=nil, deny_in_modelines=nil, normal_dname_chars=nil,
-- modelineexpr=nil,
-- expand=nil, nodefault=nil, no_mkrc=nil, vi_def=true, vim=true, -- expand=nil, nodefault=nil, no_mkrc=nil, vi_def=true, vim=true,
-- alloced=nil, -- alloced=nil,
-- save_pv_indir=nil, -- save_pv_indir=nil,
@ -283,6 +284,7 @@ return {
deny_duplicates=true, deny_duplicates=true,
vi_def=true, vi_def=true,
expand=true, expand=true,
secure=true,
varname='p_cdpath', varname='p_cdpath',
defaults={if_true={vi=",,"}} defaults={if_true={vi=",,"}}
}, },
@ -847,6 +849,7 @@ return {
type='string', scope={'window'}, type='string', scope={'window'},
vi_def=true, vi_def=true,
vim=true, vim=true,
modelineexpr=true,
alloced=true, alloced=true,
redraw={'current_window'}, redraw={'current_window'},
defaults={if_true={vi="0"}} defaults={if_true={vi="0"}}
@ -922,6 +925,7 @@ return {
type='string', scope={'window'}, type='string', scope={'window'},
vi_def=true, vi_def=true,
vim=true, vim=true,
modelineexpr=true,
alloced=true, alloced=true,
redraw={'current_window'}, redraw={'current_window'},
defaults={if_true={vi="foldtext()"}} defaults={if_true={vi="foldtext()"}}
@ -931,6 +935,7 @@ return {
type='string', scope={'buffer'}, type='string', scope={'buffer'},
vi_def=true, vi_def=true,
vim=true, vim=true,
modelineexpr=true,
alloced=true, alloced=true,
varname='p_fex', varname='p_fex',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -1045,6 +1050,7 @@ return {
full_name='guitablabel', abbreviation='gtl', full_name='guitablabel', abbreviation='gtl',
type='string', scope={'global'}, type='string', scope={'global'},
vi_def=true, vi_def=true,
modelineexpr=true,
redraw={'current_window'}, redraw={'current_window'},
enable_if=false, enable_if=false,
}, },
@ -1136,6 +1142,7 @@ return {
full_name='iconstring', full_name='iconstring',
type='string', scope={'global'}, type='string', scope={'global'},
vi_def=true, vi_def=true,
modelineexpr=true,
varname='p_iconstring', varname='p_iconstring',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
}, },
@ -1198,6 +1205,7 @@ return {
full_name='includeexpr', abbreviation='inex', full_name='includeexpr', abbreviation='inex',
type='string', scope={'buffer'}, type='string', scope={'buffer'},
vi_def=true, vi_def=true,
modelineexpr=true,
alloced=true, alloced=true,
varname='p_inex', varname='p_inex',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -1214,6 +1222,7 @@ return {
type='string', scope={'buffer'}, type='string', scope={'buffer'},
vi_def=true, vi_def=true,
vim=true, vim=true,
modelineexpr=true,
alloced=true, alloced=true,
varname='p_inde', varname='p_inde',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -1527,6 +1536,13 @@ return {
varname='p_ml', varname='p_ml',
defaults={if_true={vi=false, vim=true}} defaults={if_true={vi=false, vim=true}}
}, },
{
full_name='modelineexpr', abbreviation='mle',
type='bool', scope={'global'},
vi_def=true,
varname='p_mle',
defaults={if_true={vi=false}}
},
{ {
full_name='modelines', abbreviation='mls', full_name='modelines', abbreviation='mls',
type='number', scope={'global'}, type='number', scope={'global'},
@ -1903,6 +1919,7 @@ return {
type='string', scope={'global'}, type='string', scope={'global'},
vi_def=true, vi_def=true,
alloced=true, alloced=true,
modelineexpr=true,
redraw={'statuslines'}, redraw={'statuslines'},
varname='p_ruf', varname='p_ruf',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -2310,6 +2327,7 @@ return {
type='string', scope={'global', 'window'}, type='string', scope={'global', 'window'},
vi_def=true, vi_def=true,
alloced=true, alloced=true,
modelineexpr=true,
redraw={'statuslines'}, redraw={'statuslines'},
varname='p_stl', varname='p_stl',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -2369,6 +2387,7 @@ return {
full_name='tabline', abbreviation='tal', full_name='tabline', abbreviation='tal',
type='string', scope={'global'}, type='string', scope={'global'},
vi_def=true, vi_def=true,
modelineexpr=true,
redraw={'all_windows'}, redraw={'all_windows'},
varname='p_tal', varname='p_tal',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
@ -2528,6 +2547,7 @@ return {
full_name='titlestring', full_name='titlestring',
type='string', scope={'global'}, type='string', scope={'global'},
vi_def=true, vi_def=true,
modelineexpr=true,
varname='p_titlestring', varname='p_titlestring',
defaults={if_true={vi=""}} defaults={if_true={vi=""}}
}, },

View File

@ -4,7 +4,7 @@ If after adding a new test, the test output doesn't appear properly in
test49.failed, try to add one or more "G"s at the line ending in "test.out" test49.failed, try to add one or more "G"s at the line ending in "test.out"
STARTTEST STARTTEST
:se nomore :se nomore modelineexpr
:lang mess C :lang mess C
:so test49.vim :so test49.vim
:" Go back to this file and append the results from register r. :" Go back to this file and append the results from register r.

View File

@ -60,14 +60,17 @@ func Test_modeline_keymap()
set keymap= iminsert=0 imsearch=-1 set keymap= iminsert=0 imsearch=-1
endfunc endfunc
func s:modeline_fails(what, text) func s:modeline_fails(what, text, error)
if !exists('+' . a:what)
return
endif
let fname = "Xmodeline_fails_" . a:what let fname = "Xmodeline_fails_" . a:what
call writefile(['vim: set ' . a:text . ' :', 'nothing'], fname) call writefile(['vim: set ' . a:text . ' :', 'nothing'], fname)
let modeline = &modeline let modeline = &modeline
set modeline set modeline
filetype plugin on filetype plugin on
syntax enable syntax enable
call assert_fails('split ' . fname, 'E474:') call assert_fails('split ' . fname, a:error)
call assert_equal("", &filetype) call assert_equal("", &filetype)
call assert_equal("", &syntax) call assert_equal("", &syntax)
@ -79,16 +82,91 @@ func s:modeline_fails(what, text)
endfunc endfunc
func Test_modeline_filetype_fails() func Test_modeline_filetype_fails()
call s:modeline_fails('filetype', 'ft=evil$CMD') call s:modeline_fails('filetype', 'ft=evil$CMD', 'E474:')
endfunc endfunc
func Test_modeline_syntax_fails() func Test_modeline_syntax_fails()
call s:modeline_fails('syntax', 'syn=evil$CMD') call s:modeline_fails('syntax', 'syn=evil$CMD', 'E474:')
endfunc endfunc
func Test_modeline_keymap_fails() func Test_modeline_keymap_fails()
if !has('keymap') call s:modeline_fails('keymap', 'keymap=evil$CMD', 'E474:')
return endfunc
endif
call s:modeline_fails('keymap', 'keymap=evil$CMD') func Test_modeline_fails_always()
call s:modeline_fails('backupdir', 'backupdir=Something()', 'E520:')
call s:modeline_fails('cdpath', 'cdpath=Something()', 'E520:')
call s:modeline_fails('charconvert', 'charconvert=Something()', 'E520:')
call s:modeline_fails('completefunc', 'completefunc=Something()', 'E520:')
call s:modeline_fails('cscopeprg', 'cscopeprg=Something()', 'E520:')
call s:modeline_fails('diffexpr', 'diffexpr=Something()', 'E520:')
call s:modeline_fails('directory', 'directory=Something()', 'E520:')
call s:modeline_fails('equalprg', 'equalprg=Something()', 'E520:')
call s:modeline_fails('errorfile', 'errorfile=Something()', 'E520:')
call s:modeline_fails('exrc', 'exrc=Something()', 'E520:')
call s:modeline_fails('formatprg', 'formatprg=Something()', 'E520:')
call s:modeline_fails('fsync', 'fsync=Something()', 'E520:')
call s:modeline_fails('grepprg', 'grepprg=Something()', 'E520:')
call s:modeline_fails('helpfile', 'helpfile=Something()', 'E520:')
call s:modeline_fails('imactivatefunc', 'imactivatefunc=Something()', 'E520:')
call s:modeline_fails('imstatusfunc', 'imstatusfunc=Something()', 'E520:')
call s:modeline_fails('imstyle', 'imstyle=Something()', 'E520:')
call s:modeline_fails('keywordprg', 'keywordprg=Something()', 'E520:')
call s:modeline_fails('langmap', 'langmap=Something()', 'E520:')
call s:modeline_fails('luadll', 'luadll=Something()', 'E520:')
call s:modeline_fails('makeef', 'makeef=Something()', 'E520:')
call s:modeline_fails('makeprg', 'makeprg=Something()', 'E520:')
call s:modeline_fails('makespellmem', 'makespellmem=Something()', 'E520:')
call s:modeline_fails('mzschemedll', 'mzschemedll=Something()', 'E520:')
call s:modeline_fails('mzschemegcdll', 'mzschemegcdll=Something()', 'E520:')
call s:modeline_fails('omnifunc', 'omnifunc=Something()', 'E520:')
call s:modeline_fails('operatorfunc', 'operatorfunc=Something()', 'E520:')
call s:modeline_fails('perldll', 'perldll=Something()', 'E520:')
call s:modeline_fails('printdevice', 'printdevice=Something()', 'E520:')
call s:modeline_fails('patchexpr', 'patchexpr=Something()', 'E520:')
call s:modeline_fails('printexpr', 'printexpr=Something()', 'E520:')
call s:modeline_fails('pythondll', 'pythondll=Something()', 'E520:')
call s:modeline_fails('pythonhome', 'pythondll=Something()', 'E520:')
call s:modeline_fails('pythonthreedll', 'pythonthreedll=Something()', 'E520:')
call s:modeline_fails('pythonthreehome', 'pythonthreehome=Something()', 'E520:')
call s:modeline_fails('pyxversion', 'pyxversion=Something()', 'E520:')
call s:modeline_fails('rubydll', 'rubydll=Something()', 'E520:')
call s:modeline_fails('runtimepath', 'runtimepath=Something()', 'E520:')
call s:modeline_fails('secure', 'secure=Something()', 'E520:')
call s:modeline_fails('shell', 'shell=Something()', 'E520:')
call s:modeline_fails('shellcmdflag', 'shellcmdflag=Something()', 'E520:')
call s:modeline_fails('shellpipe', 'shellpipe=Something()', 'E520:')
call s:modeline_fails('shellquote', 'shellquote=Something()', 'E520:')
call s:modeline_fails('shellredir', 'shellredir=Something()', 'E520:')
call s:modeline_fails('shellxquote', 'shellxquote=Something()', 'E520:')
call s:modeline_fails('spellfile', 'spellfile=Something()', 'E520:')
call s:modeline_fails('spellsuggest', 'spellsuggest=Something()', 'E520:')
call s:modeline_fails('tcldll', 'tcldll=Something()', 'E520:')
call s:modeline_fails('titleold', 'titleold=Something()', 'E520:')
call s:modeline_fails('viewdir', 'viewdir=Something()', 'E520:')
call s:modeline_fails('viminfo', 'viminfo=Something()', 'E520:')
call s:modeline_fails('viminfofile', 'viminfofile=Something()', 'E520:')
call s:modeline_fails('winptydll', 'winptydll=Something()', 'E520:')
call s:modeline_fails('undodir', 'undodir=Something()', 'E520:')
" only check a few terminal options
" Skip these since nvim doesn't support termcodes as options
"call s:modeline_fails('t_AB', 't_AB=Something()', 'E520:')
"call s:modeline_fails('t_ce', 't_ce=Something()', 'E520:')
"call s:modeline_fails('t_sr', 't_sr=Something()', 'E520:')
"call s:modeline_fails('t_8b', 't_8b=Something()', 'E520:')
endfunc
func Test_modeline_fails_modelineexpr()
call s:modeline_fails('balloonexpr', 'balloonexpr=Something()', 'E992:')
call s:modeline_fails('foldexpr', 'foldexpr=Something()', 'E992:')
call s:modeline_fails('foldtext', 'foldtext=Something()', 'E992:')
call s:modeline_fails('formatexpr', 'formatexpr=Something()', 'E992:')
call s:modeline_fails('guitablabel', 'guitablabel=Something()', 'E992:')
call s:modeline_fails('iconstring', 'iconstring=Something()', 'E992:')
call s:modeline_fails('includeexpr', 'includeexpr=Something()', 'E992:')
call s:modeline_fails('indentexpr', 'indentexpr=Something()', 'E992:')
call s:modeline_fails('rulerformat', 'rulerformat=Something()', 'E992:')
call s:modeline_fails('statusline', 'statusline=Something()', 'E992:')
call s:modeline_fails('tabline', 'tabline=Something()', 'E992:')
call s:modeline_fails('titlestring', 'titlestring=Something()', 'E992:')
endfunc endfunc