Bug fix: proper escaping in config files.
This commit is contained in:
parent
aa73921d30
commit
45f9e93b00
@ -15,41 +15,28 @@ function! syntastic#c#NullOutput()
|
|||||||
return known_os ? '-o ' . syntastic#util#DevNull() : ''
|
return known_os ? '-o ' . syntastic#util#DevNull() : ''
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
" get the gcc include directory argument depending on the default
|
|
||||||
" includes and the optional user-defined 'g:syntastic_c_include_dirs'
|
|
||||||
function! syntastic#c#GetIncludeDirs(filetype)
|
|
||||||
let include_dirs = []
|
|
||||||
|
|
||||||
if !exists('g:syntastic_'.a:filetype.'_no_default_include_dirs') ||
|
|
||||||
\ !g:syntastic_{a:filetype}_no_default_include_dirs
|
|
||||||
let include_dirs = copy(s:default_includes)
|
|
||||||
endif
|
|
||||||
|
|
||||||
if exists('g:syntastic_'.a:filetype.'_include_dirs')
|
|
||||||
call extend(include_dirs, g:syntastic_{a:filetype}_include_dirs)
|
|
||||||
endif
|
|
||||||
|
|
||||||
return join(map(syntastic#util#unique(include_dirs), '"-I" . v:val'), ' ')
|
|
||||||
endfunction
|
|
||||||
|
|
||||||
" read additional compiler flags from the given configuration file
|
" read additional compiler flags from the given configuration file
|
||||||
" the file format and its parsing mechanism is inspired by clang_complete
|
" the file format and its parsing mechanism is inspired by clang_complete
|
||||||
function! syntastic#c#ReadConfig(file)
|
function! syntastic#c#ReadConfig(file)
|
||||||
" search in the current file's directory upwards
|
" search in the current file's directory upwards
|
||||||
let config = findfile(a:file, '.;')
|
let config = findfile(a:file, '.;')
|
||||||
if config == '' || !filereadable(config) | return '' | endif
|
if config == '' || !filereadable(config)
|
||||||
|
return ''
|
||||||
|
endif
|
||||||
|
|
||||||
" convert filename into absolute path
|
" convert filename into absolute path
|
||||||
let filepath = substitute(fnamemodify(config, ':p:h'), '\', '/', 'g')
|
let filepath = fnamemodify(config, ':p:h')
|
||||||
|
|
||||||
" try to read config file
|
" try to read config file
|
||||||
try
|
try
|
||||||
let lines = map(readfile(config),
|
let lines = readfile(config)
|
||||||
\ 'substitute(v:val, ''\'', ''/'', ''g'')')
|
catch /^Vim\%((\a\+)\)\=:E484/
|
||||||
catch /E484/
|
|
||||||
return ''
|
return ''
|
||||||
endtry
|
endtry
|
||||||
|
|
||||||
|
" filter out empty lines and comments
|
||||||
|
call filter(lines, 'v:val !~ ''\v^(\s*#|$)''')
|
||||||
|
|
||||||
let parameters = []
|
let parameters = []
|
||||||
for line in lines
|
for line in lines
|
||||||
let matches = matchlist(line, '\C^\s*-I\s*\(\S\+\)')
|
let matches = matchlist(line, '\C^\s*-I\s*\(\S\+\)')
|
||||||
@ -58,67 +45,14 @@ function! syntastic#c#ReadConfig(file)
|
|||||||
if match(matches[1], '^\%(/\|\a:\)') != -1
|
if match(matches[1], '^\%(/\|\a:\)') != -1
|
||||||
call add(parameters, '-I' . matches[1])
|
call add(parameters, '-I' . matches[1])
|
||||||
else
|
else
|
||||||
call add(parameters, '-I' . filepath . '/' . matches[1])
|
call add(parameters, '-I' . filepath . syntastic#util#Slash() . matches[1])
|
||||||
endif
|
endif
|
||||||
else
|
else
|
||||||
call add(parameters, line)
|
call add(parameters, line)
|
||||||
endif
|
endif
|
||||||
endfor
|
endfor
|
||||||
|
|
||||||
return join(parameters, ' ')
|
return join(map(parameters, 'shellescape(fnameescape(v:val))'), ' ')
|
||||||
endfunction
|
|
||||||
|
|
||||||
" search the first 100 lines for include statements that are
|
|
||||||
" given in the handlers dictionary
|
|
||||||
function! syntastic#c#SearchHeaders()
|
|
||||||
let includes = ''
|
|
||||||
let files = []
|
|
||||||
let found = []
|
|
||||||
let lines = filter(getline(1, 100), 'v:val =~# "^\s*#\s*include"')
|
|
||||||
|
|
||||||
" search current buffer
|
|
||||||
for line in lines
|
|
||||||
let file = matchstr(line, '"\zs\S\+\ze"')
|
|
||||||
if file != ''
|
|
||||||
call add(files, file)
|
|
||||||
continue
|
|
||||||
endif
|
|
||||||
for handler in s:handlers
|
|
||||||
if line =~# handler["regex"]
|
|
||||||
let includes .= call(handler["func"], handler["args"])
|
|
||||||
call add(found, handler["regex"])
|
|
||||||
break
|
|
||||||
endif
|
|
||||||
endfor
|
|
||||||
endfor
|
|
||||||
|
|
||||||
" search included headers
|
|
||||||
for hfile in files
|
|
||||||
if hfile != ''
|
|
||||||
let filename = expand('%:p:h') . (has('win32') ?
|
|
||||||
\ '\' : '/') . hfile
|
|
||||||
try
|
|
||||||
let lines = readfile(filename, '', 100)
|
|
||||||
catch /E484/
|
|
||||||
continue
|
|
||||||
endtry
|
|
||||||
let lines = filter(lines, 'v:val =~# "^\s*#\s*include"')
|
|
||||||
for handler in s:handlers
|
|
||||||
if index(found, handler["regex"]) != -1
|
|
||||||
continue
|
|
||||||
endif
|
|
||||||
for line in lines
|
|
||||||
if line =~# handler["regex"]
|
|
||||||
let includes .= call(handler["func"], handler["args"])
|
|
||||||
call add(found, handler["regex"])
|
|
||||||
break
|
|
||||||
endif
|
|
||||||
endfor
|
|
||||||
endfor
|
|
||||||
endif
|
|
||||||
endfor
|
|
||||||
|
|
||||||
return includes
|
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
" GetLocList() for C-like compilers
|
" GetLocList() for C-like compilers
|
||||||
@ -134,7 +68,7 @@ function! syntastic#c#GetLocList(filetype, options)
|
|||||||
\ g:syntastic_{ft}_compiler .
|
\ g:syntastic_{ft}_compiler .
|
||||||
\ ' ' . get(a:options, 'makeprg_headers', '') .
|
\ ' ' . get(a:options, 'makeprg_headers', '') .
|
||||||
\ ' ' . g:syntastic_{ft}_compiler_options .
|
\ ' ' . g:syntastic_{ft}_compiler_options .
|
||||||
\ ' ' . syntastic#c#GetIncludeDirs(ft) .
|
\ ' ' . s:GetIncludeDirs(ft) .
|
||||||
\ ' ' . syntastic#c#NullOutput() .
|
\ ' ' . syntastic#c#NullOutput() .
|
||||||
\ ' -c ' . shellescape(expand('%'))
|
\ ' -c ' . shellescape(expand('%'))
|
||||||
else
|
else
|
||||||
@ -145,7 +79,7 @@ function! syntastic#c#GetLocList(filetype, options)
|
|||||||
\ g:syntastic_{ft}_compiler .
|
\ g:syntastic_{ft}_compiler .
|
||||||
\ ' ' . get(a:options, 'makeprg_main', '') .
|
\ ' ' . get(a:options, 'makeprg_main', '') .
|
||||||
\ ' ' . g:syntastic_{ft}_compiler_options .
|
\ ' ' . g:syntastic_{ft}_compiler_options .
|
||||||
\ ' ' . syntastic#c#GetIncludeDirs(ft) .
|
\ ' ' . s:GetIncludeDirs(ft) .
|
||||||
\ ' ' . shellescape(expand('%'))
|
\ ' ' . shellescape(expand('%'))
|
||||||
endif
|
endif
|
||||||
|
|
||||||
@ -156,11 +90,11 @@ function! syntastic#c#GetLocList(filetype, options)
|
|||||||
if ft ==# 'c' || ft ==# 'cpp'
|
if ft ==# 'c' || ft ==# 'cpp'
|
||||||
" refresh the include file search if desired
|
" refresh the include file search if desired
|
||||||
if exists('g:syntastic_' . ft . '_auto_refresh_includes') && g:syntastic_{ft}_auto_refresh_includes
|
if exists('g:syntastic_' . ft . '_auto_refresh_includes') && g:syntastic_{ft}_auto_refresh_includes
|
||||||
let makeprg .= ' ' . syntastic#c#SearchHeaders()
|
let makeprg .= ' ' . s:SearchHeaders()
|
||||||
else
|
else
|
||||||
" search for header includes if not cached already
|
" search for header includes if not cached already
|
||||||
if !exists('b:syntastic_' . ft . '_includes')
|
if !exists('b:syntastic_' . ft . '_includes')
|
||||||
let b:syntastic_{ft}_includes = syntastic#c#SearchHeaders()
|
let b:syntastic_{ft}_includes = s:SearchHeaders()
|
||||||
endif
|
endif
|
||||||
let makeprg .= ' ' . b:syntastic_{ft}_includes
|
let makeprg .= ' ' . b:syntastic_{ft}_includes
|
||||||
endif
|
endif
|
||||||
@ -192,31 +126,94 @@ function! s:Init()
|
|||||||
let s:handlers = []
|
let s:handlers = []
|
||||||
let s:cflags = {}
|
let s:cflags = {}
|
||||||
|
|
||||||
call s:RegHandler('gtk', 'syntastic#c#CheckPKG',
|
call s:RegHandler('cairo', 'syntastic#c#CheckPKG', ['cairo', 'cairo'])
|
||||||
\ ['gtk', 'gtk+-2.0', 'gtk+', 'glib-2.0', 'glib'])
|
call s:RegHandler('freetype', 'syntastic#c#CheckPKG', ['freetype', 'freetype2', 'freetype'])
|
||||||
call s:RegHandler('glib', 'syntastic#c#CheckPKG',
|
call s:RegHandler('glade', 'syntastic#c#CheckPKG', ['glade', 'libglade-2.0', 'libglade'])
|
||||||
\ ['glib', 'glib-2.0', 'glib'])
|
call s:RegHandler('glib', 'syntastic#c#CheckPKG', ['glib', 'glib-2.0', 'glib'])
|
||||||
call s:RegHandler('glade', 'syntastic#c#CheckPKG',
|
call s:RegHandler('gtk', 'syntastic#c#CheckPKG', ['gtk', 'gtk+-2.0', 'gtk+', 'glib-2.0', 'glib'])
|
||||||
\ ['glade', 'libglade-2.0', 'libglade'])
|
call s:RegHandler('libsoup', 'syntastic#c#CheckPKG', ['libsoup', 'libsoup-2.4', 'libsoup-2.2'])
|
||||||
call s:RegHandler('libsoup', 'syntastic#c#CheckPKG',
|
call s:RegHandler('libxml', 'syntastic#c#CheckPKG', ['libxml', 'libxml-2.0', 'libxml'])
|
||||||
\ ['libsoup', 'libsoup-2.4', 'libsoup-2.2'])
|
call s:RegHandler('pango', 'syntastic#c#CheckPKG', ['pango', 'pango'])
|
||||||
call s:RegHandler('webkit', 'syntastic#c#CheckPKG',
|
call s:RegHandler('SDL', 'syntastic#c#CheckPKG', ['sdl', 'sdl'])
|
||||||
\ ['webkit', 'webkit-1.0'])
|
call s:RegHandler('opengl', 'syntastic#c#CheckPKG', ['opengl', 'gl'])
|
||||||
call s:RegHandler('cairo', 'syntastic#c#CheckPKG',
|
call s:RegHandler('webkit', 'syntastic#c#CheckPKG', ['webkit', 'webkit-1.0'])
|
||||||
\ ['cairo', 'cairo'])
|
|
||||||
call s:RegHandler('pango', 'syntastic#c#CheckPKG',
|
call s:RegHandler('php\.h', 'syntastic#c#CheckPhp', [])
|
||||||
\ ['pango', 'pango'])
|
|
||||||
call s:RegHandler('libxml', 'syntastic#c#CheckPKG',
|
|
||||||
\ ['libxml', 'libxml-2.0', 'libxml'])
|
|
||||||
call s:RegHandler('freetype', 'syntastic#c#CheckPKG',
|
|
||||||
\ ['freetype', 'freetype2', 'freetype'])
|
|
||||||
call s:RegHandler('SDL', 'syntastic#c#CheckPKG',
|
|
||||||
\ ['sdl', 'sdl'])
|
|
||||||
call s:RegHandler('opengl', 'syntastic#c#CheckPKG',
|
|
||||||
\ ['opengl', 'gl'])
|
|
||||||
call s:RegHandler('ruby', 'syntastic#c#CheckRuby', [])
|
|
||||||
call s:RegHandler('Python\.h', 'syntastic#c#CheckPython', [])
|
call s:RegHandler('Python\.h', 'syntastic#c#CheckPython', [])
|
||||||
call s:RegHandler('php\.h', 'syntastic#c#CheckPhp', [])
|
call s:RegHandler('ruby', 'syntastic#c#CheckRuby', [])
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
" get the gcc include directory argument depending on the default
|
||||||
|
" includes and the optional user-defined 'g:syntastic_c_include_dirs'
|
||||||
|
function! s:GetIncludeDirs(filetype)
|
||||||
|
let include_dirs = []
|
||||||
|
|
||||||
|
if !exists('g:syntastic_'.a:filetype.'_no_default_include_dirs') || !g:syntastic_{a:filetype}_no_default_include_dirs
|
||||||
|
let include_dirs = copy(s:default_includes)
|
||||||
|
endif
|
||||||
|
|
||||||
|
if exists('g:syntastic_'.a:filetype.'_include_dirs')
|
||||||
|
call extend(include_dirs, g:syntastic_{a:filetype}_include_dirs)
|
||||||
|
endif
|
||||||
|
|
||||||
|
return join(map(syntastic#util#unique(include_dirs), 'shellescape(fnameescape("-I" . v:val))'), ' ')
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
" search the first 100 lines for include statements that are
|
||||||
|
" given in the handlers dictionary
|
||||||
|
function! s:SearchHeaders()
|
||||||
|
let includes = ''
|
||||||
|
let files = []
|
||||||
|
let found = []
|
||||||
|
let lines = filter(getline(1, 100), 'v:val =~# "^\s*#\s*include"')
|
||||||
|
|
||||||
|
" search current buffer
|
||||||
|
for line in lines
|
||||||
|
let file = matchstr(line, '"\zs\S\+\ze"')
|
||||||
|
if file != ''
|
||||||
|
call add(files, file)
|
||||||
|
continue
|
||||||
|
endif
|
||||||
|
|
||||||
|
for handler in s:handlers
|
||||||
|
if line =~# handler["regex"]
|
||||||
|
let includes .= call(handler["func"], handler["args"])
|
||||||
|
call add(found, handler["regex"])
|
||||||
|
break
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
|
||||||
|
" search included headers
|
||||||
|
for hfile in files
|
||||||
|
if hfile != ''
|
||||||
|
let filename = expand('%:p:h') . syntastic#util#Slash() . hfile
|
||||||
|
|
||||||
|
try
|
||||||
|
let lines = readfile(filename, '', 100)
|
||||||
|
catch /^Vim\%((\a\+)\)\=:E484/
|
||||||
|
continue
|
||||||
|
endtry
|
||||||
|
|
||||||
|
let lines = filter(lines, 'v:val =~# "^\s*#\s*include"')
|
||||||
|
|
||||||
|
for handler in s:handlers
|
||||||
|
if index(found, handler["regex"]) != -1
|
||||||
|
continue
|
||||||
|
endif
|
||||||
|
|
||||||
|
for line in lines
|
||||||
|
if line =~# handler["regex"]
|
||||||
|
let includes .= call(handler["func"], handler["args"])
|
||||||
|
call add(found, handler["regex"])
|
||||||
|
break
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
|
||||||
|
return includes
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
" try to find library with 'pkg-config'
|
" try to find library with 'pkg-config'
|
||||||
@ -225,14 +222,14 @@ endfunction
|
|||||||
function! syntastic#c#CheckPKG(name, ...)
|
function! syntastic#c#CheckPKG(name, ...)
|
||||||
if executable('pkg-config')
|
if executable('pkg-config')
|
||||||
if !has_key(s:cflags, a:name)
|
if !has_key(s:cflags, a:name)
|
||||||
for i in range(a:0)
|
for pkg in a:000
|
||||||
let l:cflags = system('pkg-config --cflags '.a:000[i])
|
let pkg_flags = system('pkg-config --cflags ' . pkg)
|
||||||
" since we cannot necessarily trust the pkg-config exit code
|
" since we cannot necessarily trust the pkg-config exit code
|
||||||
" we have to check for an error output as well
|
" we have to check for an error output as well
|
||||||
if v:shell_error == 0 && l:cflags !~? 'not found'
|
if v:shell_error == 0 && pkg_flags !~? 'not found'
|
||||||
let l:cflags = ' '.substitute(l:cflags, "\n", '', '')
|
let pkg_flags = ' ' . substitute(pkg_flags, "\n", '', '')
|
||||||
let s:cflags[a:name] = l:cflags
|
let s:cflags[a:name] = pkg_flags
|
||||||
return l:cflags
|
return pkg_flags
|
||||||
endif
|
endif
|
||||||
endfor
|
endfor
|
||||||
else
|
else
|
||||||
@ -245,11 +242,11 @@ endfunction
|
|||||||
" try to find PHP includes with 'php-config'
|
" try to find PHP includes with 'php-config'
|
||||||
function! syntastic#c#CheckPhp()
|
function! syntastic#c#CheckPhp()
|
||||||
if executable('php-config')
|
if executable('php-config')
|
||||||
if !exists('s:php_flags')
|
if !has_key(s:cflags, 'php')
|
||||||
let s:php_flags = system('php-config --includes')
|
let s:cflags['php'] = system('php-config --includes')
|
||||||
let s:php_flags = ' ' . substitute(s:php_flags, "\n", '', '')
|
let s:cflags['php'] = ' ' . substitute(s:cflags['php'], "\n", '', '')
|
||||||
endif
|
endif
|
||||||
return s:php_flags
|
return s:cflags['php']
|
||||||
endif
|
endif
|
||||||
return ''
|
return ''
|
||||||
endfunction
|
endfunction
|
||||||
@ -257,13 +254,12 @@ endfunction
|
|||||||
" try to find the ruby headers with 'rbconfig'
|
" try to find the ruby headers with 'rbconfig'
|
||||||
function! syntastic#c#CheckRuby()
|
function! syntastic#c#CheckRuby()
|
||||||
if executable('ruby')
|
if executable('ruby')
|
||||||
if !exists('s:ruby_flags')
|
if !has_key(s:cflags, 'ruby')
|
||||||
let s:ruby_flags = system('ruby -r rbconfig -e '
|
let s:cflags['ruby'] = system('ruby -r rbconfig -e ''puts Config::CONFIG["archdir"]''')
|
||||||
\ . '''puts Config::CONFIG["archdir"]''')
|
let s:cflags['ruby'] = substitute(s:cflags['ruby'], "\n", '', '')
|
||||||
let s:ruby_flags = substitute(s:ruby_flags, "\n", '', '')
|
let s:cflags['ruby'] = ' -I' . s:cflags['ruby']
|
||||||
let s:ruby_flags = ' -I' . s:ruby_flags
|
|
||||||
endif
|
endif
|
||||||
return s:ruby_flags
|
return s:cflags['ruby']
|
||||||
endif
|
endif
|
||||||
return ''
|
return ''
|
||||||
endfunction
|
endfunction
|
||||||
@ -271,13 +267,13 @@ endfunction
|
|||||||
" try to find the python headers with distutils
|
" try to find the python headers with distutils
|
||||||
function! syntastic#c#CheckPython()
|
function! syntastic#c#CheckPython()
|
||||||
if executable('python')
|
if executable('python')
|
||||||
if !exists('s:python_flags')
|
if !has_key(s:cflags, 'python')
|
||||||
let s:python_flags = system('python -c ''from distutils import '
|
let s:cflags['python'] = system('python -c ''from distutils import ' .
|
||||||
\ . 'sysconfig; import sys; sys.stdout.write(sysconfig.get_python_inc())''')
|
\ 'sysconfig; import sys; sys.stdout.write(sysconfig.get_python_inc())''')
|
||||||
let s:python_flags = substitute(s:python_flags, "\n", '', '')
|
let s:cflags['python'] = substitute(s:cflags['python'], "\n", '', '')
|
||||||
let s:python_flags = ' -I' . s:python_flags
|
let s:cflags['python'] = ' -I' . s:cflags['python']
|
||||||
endif
|
endif
|
||||||
return s:python_flags
|
return s:cflags['python']
|
||||||
endif
|
endif
|
||||||
return ''
|
return ''
|
||||||
endfunction
|
endfunction
|
||||||
@ -294,8 +290,13 @@ endfunction
|
|||||||
" }}}1
|
" }}}1
|
||||||
|
|
||||||
" default include directories
|
" default include directories
|
||||||
let s:default_includes = [ '.', '..', 'include', 'includes',
|
let s:default_includes = [
|
||||||
\ '../include', '../includes' ]
|
\ '.',
|
||||||
|
\ '..',
|
||||||
|
\ 'include',
|
||||||
|
\ 'includes',
|
||||||
|
\ '..' . syntastic#util#Slash() . 'include',
|
||||||
|
\ '..' . syntastic#util#Slash() . 'includes' ]
|
||||||
|
|
||||||
call s:Init()
|
call s:Init()
|
||||||
|
|
||||||
|
@ -19,6 +19,11 @@ function! syntastic#util#DevNull()
|
|||||||
return '/dev/null'
|
return '/dev/null'
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
|
" Get directory separator
|
||||||
|
function! syntastic#util#Slash() abort
|
||||||
|
return !exists("+shellslash") || &shellslash ? '/' : '\'
|
||||||
|
endfunction
|
||||||
|
|
||||||
"search the first 5 lines of the file for a magic number and return a map
|
"search the first 5 lines of the file for a magic number and return a map
|
||||||
"containing the args and the executable
|
"containing the args and the executable
|
||||||
"
|
"
|
||||||
|
Loading…
Reference in New Issue
Block a user