2017-02-20 21:17:09 +08:00
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import fnmatch
|
|
|
|
import logging
|
|
|
|
import ycm_core
|
|
|
|
|
|
|
|
BASE_FLAGS = [
|
|
|
|
'-Wall',
|
|
|
|
'-Wextra',
|
|
|
|
'-Werror',
|
|
|
|
'-Wno-long-long',
|
|
|
|
'-Wno-variadic-macros',
|
|
|
|
'-fexceptions',
|
|
|
|
'-ferror-limit=10000',
|
|
|
|
'-DNDEBUG',
|
2017-02-22 18:08:17 +08:00
|
|
|
'-std=c++1z',
|
2017-02-20 21:17:09 +08:00
|
|
|
'-xc++',
|
|
|
|
'-I/usr/lib/',
|
|
|
|
'-I/usr/include/',
|
|
|
|
'-I./src',
|
|
|
|
'-I./include',
|
|
|
|
'-I./libs/fmt',
|
|
|
|
'-I./libs/yaml-cpp',
|
2017-06-21 17:29:13 +08:00
|
|
|
'-I./libs/glog/include',
|
2017-02-24 00:15:31 +08:00
|
|
|
'-I./libs/googletest/googletest/include',
|
|
|
|
'-I./libs/googletest/googlemock/include',
|
|
|
|
'-I./libs/benchmark/include',
|
2017-07-12 18:44:21 +08:00
|
|
|
# We include cppitertools headers directly from libs directory.
|
|
|
|
'-I./libs',
|
2017-07-06 19:53:39 +08:00
|
|
|
'-I./libs/rapidcheck/include',
|
2017-06-07 16:15:08 +08:00
|
|
|
'-I./libs/antlr4/runtime/Cpp/runtime/src',
|
|
|
|
'-I./build/libs/gflags/include'
|
2017-02-20 21:17:09 +08:00
|
|
|
]
|
|
|
|
|
|
|
|
SOURCE_EXTENSIONS = [
|
|
|
|
'.cpp',
|
|
|
|
'.cxx',
|
|
|
|
'.cc',
|
|
|
|
'.c',
|
|
|
|
'.m',
|
|
|
|
'.mm'
|
|
|
|
]
|
|
|
|
|
|
|
|
HEADER_EXTENSIONS = [
|
|
|
|
'.h',
|
|
|
|
'.hxx',
|
|
|
|
'.hpp',
|
|
|
|
'.hh'
|
|
|
|
]
|
|
|
|
|
|
|
|
def IsHeaderFile(filename):
|
|
|
|
extension = os.path.splitext(filename)[1]
|
|
|
|
return extension in HEADER_EXTENSIONS
|
|
|
|
|
|
|
|
def GetCompilationInfoForFile(database, filename):
|
|
|
|
if IsHeaderFile(filename):
|
|
|
|
basename = os.path.splitext(filename)[0]
|
|
|
|
for extension in SOURCE_EXTENSIONS:
|
|
|
|
replacement_file = basename + extension
|
|
|
|
if os.path.exists(replacement_file):
|
|
|
|
compilation_info = database.GetCompilationInfoForFile(replacement_file)
|
|
|
|
if compilation_info.compiler_flags_:
|
|
|
|
return compilation_info
|
|
|
|
return None
|
|
|
|
return database.GetCompilationInfoForFile(filename)
|
|
|
|
|
|
|
|
def FindNearest(path, target):
|
|
|
|
candidate = os.path.join(path, target)
|
|
|
|
if(os.path.isfile(candidate) or os.path.isdir(candidate)):
|
|
|
|
logging.info("Found nearest " + target + " at " + candidate)
|
|
|
|
return candidate;
|
|
|
|
else:
|
|
|
|
parent = os.path.dirname(os.path.abspath(path));
|
|
|
|
if(parent == path):
|
|
|
|
raise RuntimeError("Could not find " + target);
|
|
|
|
return FindNearest(parent, target)
|
|
|
|
|
|
|
|
def MakeRelativePathsInFlagsAbsolute(flags, working_directory):
|
|
|
|
if not working_directory:
|
|
|
|
return list(flags)
|
|
|
|
new_flags = []
|
|
|
|
make_next_absolute = False
|
|
|
|
path_flags = [ '-isystem', '-I', '-iquote', '--sysroot=' ]
|
|
|
|
for flag in flags:
|
|
|
|
new_flag = flag
|
|
|
|
|
|
|
|
if make_next_absolute:
|
|
|
|
make_next_absolute = False
|
|
|
|
if not flag.startswith('/'):
|
|
|
|
new_flag = os.path.join(working_directory, flag)
|
|
|
|
|
|
|
|
for path_flag in path_flags:
|
|
|
|
if flag == path_flag:
|
|
|
|
make_next_absolute = True
|
|
|
|
break
|
|
|
|
|
|
|
|
if flag.startswith(path_flag):
|
|
|
|
path = flag[ len(path_flag): ]
|
|
|
|
new_flag = path_flag + os.path.join(working_directory, path)
|
|
|
|
break
|
|
|
|
|
|
|
|
if new_flag:
|
|
|
|
new_flags.append(new_flag)
|
|
|
|
return new_flags
|
|
|
|
|
|
|
|
|
|
|
|
def FlagsForClangComplete(root):
|
|
|
|
try:
|
|
|
|
clang_complete_path = FindNearest(root, '.clang_complete')
|
|
|
|
clang_complete_flags = open(clang_complete_path, 'r').read().splitlines()
|
|
|
|
return clang_complete_flags
|
|
|
|
except:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def FlagsForInclude(root):
|
|
|
|
try:
|
|
|
|
include_path = FindNearest(root, 'include')
|
|
|
|
flags = []
|
|
|
|
for dirroot, dirnames, filenames in os.walk(include_path):
|
|
|
|
for dir_path in dirnames:
|
|
|
|
real_path = os.path.join(dirroot, dir_path)
|
|
|
|
flags = flags + ["-I" + real_path]
|
|
|
|
return flags
|
|
|
|
except:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def FlagsForCompilationDatabase(root, filename):
|
|
|
|
try:
|
|
|
|
compilation_db_path = FindNearest(root, 'compile_commands.json')
|
|
|
|
compilation_db_dir = os.path.dirname(compilation_db_path)
|
|
|
|
logging.info("Set compilation database directory to " + compilation_db_dir)
|
|
|
|
compilation_db = ycm_core.CompilationDatabase(compilation_db_dir)
|
|
|
|
if not compilation_db:
|
|
|
|
logging.info("Compilation database file found but unable to load")
|
|
|
|
return None
|
|
|
|
compilation_info = GetCompilationInfoForFile(compilation_db, filename)
|
|
|
|
if not compilation_info:
|
|
|
|
logging.info("No compilation info for " + filename + " in compilation database")
|
|
|
|
return None
|
|
|
|
return MakeRelativePathsInFlagsAbsolute(
|
|
|
|
compilation_info.compiler_flags_,
|
|
|
|
compilation_info.compiler_working_dir_)
|
|
|
|
except:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def FlagsForFile(filename):
|
|
|
|
root = os.path.realpath(filename);
|
|
|
|
compilation_db_flags = FlagsForCompilationDatabase(root, filename)
|
|
|
|
if compilation_db_flags:
|
|
|
|
final_flags = compilation_db_flags
|
|
|
|
else:
|
|
|
|
final_flags = BASE_FLAGS
|
|
|
|
clang_flags = FlagsForClangComplete(root)
|
|
|
|
if clang_flags:
|
|
|
|
final_flags = final_flags + clang_flags
|
|
|
|
include_flags = FlagsForInclude(root)
|
|
|
|
if include_flags:
|
|
|
|
final_flags = final_flags + include_flags
|
|
|
|
return {
|
|
|
|
'flags': final_flags,
|
|
|
|
'do_cache': True
|
|
|
|
}
|