From 49c687663e21b82bc174dd8865e1aa86696c7807 Mon Sep 17 00:00:00 2001 From: Jacob Dufault Date: Tue, 25 Apr 2017 21:03:22 -0700 Subject: [PATCH] Rework some of the command line flag parsing logic. Hopefully make it more robust. --- src/code_completion.cc | 13 +-- src/code_completion.h | 2 +- src/command_line.cc | 2 +- src/indexer.cc | 2 +- src/libclangmm/TranslationUnit.cc | 11 +- src/libclangmm/TranslationUnit.h | 3 +- src/project.cc | 183 +++++++++++------------------- src/project.h | 2 +- src/utils.cc | 12 -- src/utils.h | 20 +++- 10 files changed, 96 insertions(+), 154 deletions(-) diff --git a/src/code_completion.cc b/src/code_completion.cc index de0a57b1..4ddffc63 100644 --- a/src/code_completion.cc +++ b/src/code_completion.cc @@ -324,22 +324,17 @@ void CompletionMain(CompletionManager* completion_manager) { } // namespace -CompletionSession::CompletionSession(const Project::Entry& file, IndexerConfig* config, WorkingFiles* working_files) : file(file) { +CompletionSession::CompletionSession(const Project::Entry& file, WorkingFiles* working_files) : file(file) { std::vector unsaved = working_files->AsUnsavedFiles(); std::vector args = file.args; - args.push_back("-x"); - args.push_back("c++"); args.push_back("-fparse-all-comments"); - std::string sent_args = ""; - for (auto& arg : args) - sent_args += arg + ", "; - std::cerr << "Creating completion session with arguments " << sent_args << std::endl; + std::cerr << "Creating completion session with arguments " << StringJoin(args) << std::endl; // TODO: I think we crash when there are syntax errors. active_index = MakeUnique(0 /*excludeDeclarationsFromPCH*/, 0 /*displayDiagnostics*/); - active = MakeUnique(config, *active_index, file.filename, args, unsaved, Flags()); + active = MakeUnique(*active_index, file.filename, args, unsaved, Flags()); std::cerr << "Done creating active; did_fail=" << active->did_fail << std::endl; //if (active->did_fail) { // std::cerr << "Failed to create translation unit; trying again..." << std::endl; @@ -398,6 +393,6 @@ CompletionSession* CompletionManager::GetOrOpenSession(const std::string& filena else { std::cerr << "Found compilation entry" << std::endl; } - sessions.push_back(MakeUnique(*entry, config, working_files)); + sessions.push_back(MakeUnique(*entry, working_files)); return sessions[sessions.size() - 1].get(); } diff --git a/src/code_completion.h b/src/code_completion.h index 327d4db6..56ff02c0 100644 --- a/src/code_completion.h +++ b/src/code_completion.h @@ -23,7 +23,7 @@ struct CompletionSession { //std::unique_ptr updated; //std::unique_ptr updated_index; - CompletionSession(const Project::Entry& file, IndexerConfig* config, WorkingFiles* working_files); + CompletionSession(const Project::Entry& file, WorkingFiles* working_files); ~CompletionSession(); // Refresh file index. diff --git a/src/command_line.cc b/src/command_line.cc index 6029c162..71f340f0 100644 --- a/src/command_line.cc +++ b/src/command_line.cc @@ -1282,7 +1282,7 @@ bool QueryDbMainLoop( } // Open up / load the project. - project->Load(project_path); + project->Load(config->extraClangArguments, project_path); std::cerr << "Loaded compilation entries (" << project->entries.size() << " files)" << std::endl; project->ForAllFilteredFiles(config, [&](int i, const Project::Entry& entry) { diff --git a/src/indexer.cc b/src/indexer.cc index 4d942085..bf74e4e7 100644 --- a/src/indexer.cc +++ b/src/indexer.cc @@ -1355,7 +1355,7 @@ std::vector> Parse( clang::Index index(0 /*excludeDeclarationsFromPCH*/, 0 /*displayDiagnostics*/); std::vector unsaved_files; - clang::TranslationUnit tu(config, index, file, args, unsaved_files, CXTranslationUnit_KeepGoing); + clang::TranslationUnit tu(index, file, args, unsaved_files, CXTranslationUnit_KeepGoing); if (dump_ast) Dump(tu.document_cursor()); diff --git a/src/libclangmm/TranslationUnit.cc b/src/libclangmm/TranslationUnit.cc index 336d0a98..7579ef5b 100644 --- a/src/libclangmm/TranslationUnit.cc +++ b/src/libclangmm/TranslationUnit.cc @@ -10,8 +10,7 @@ namespace clang { -TranslationUnit::TranslationUnit(IndexerConfig* config, - Index& index, +TranslationUnit::TranslationUnit(Index& index, const std::string& filepath, const std::vector& arguments, std::vector unsaved_files, @@ -24,13 +23,7 @@ TranslationUnit::TranslationUnit(IndexerConfig* config, for (const auto& arg : platform_args) args.push_back(arg.c_str()); - for (const std::string& arg : config->extraClangArguments) - args.push_back(arg.c_str()); - - std::cerr << "Parsing " << filepath << " with args "; - for (const auto& arg : args) - std::cerr << arg << " "; - std::cerr << std::endl; + std::cerr << "Parsing " << filepath << " with args " << StringJoin(args) << std::endl; CXErrorCode error_code = clang_parseTranslationUnit2FullArgv( index.cx_index, filepath.c_str(), args.data(), args.size(), diff --git a/src/libclangmm/TranslationUnit.h b/src/libclangmm/TranslationUnit.h index 24514db5..1c89bc1a 100644 --- a/src/libclangmm/TranslationUnit.h +++ b/src/libclangmm/TranslationUnit.h @@ -12,8 +12,7 @@ namespace clang { class TranslationUnit { public: - TranslationUnit(IndexerConfig* config, - Index& index, + TranslationUnit(Index& index, const std::string& filepath, const std::vector& arguments, std::vector unsaved_files, diff --git a/src/project.cc b/src/project.cc index 790c8683..83a51a4f 100644 --- a/src/project.cc +++ b/src/project.cc @@ -11,7 +11,6 @@ #include #include - struct CompileCommandsEntry { std::string directory; std::string file; @@ -22,137 +21,83 @@ MAKE_REFLECT_STRUCT(CompileCommandsEntry, directory, file, command, args); namespace { - - -// https://github.com/Andersbakken/rtags/blob/6b16b81ea93aeff4a58930b44b2a0a207b456192/src/Source.cpp -static const char *kValueArgs[] = { - "--param", - "-G", - "-MF", - "-MQ", - "-MT", - "-T", - "-V", - "-Xanalyzer", - "-Xassembler", - "-Xclang", - "-Xlinker", - "-Xpreprocessor", - "-arch", - "-b", - "-gcc-toolchain", - //"-imacros", - "-imultilib", - //"-include", - //"-iprefix", - //"-isysroot", - "-ivfsoverlay", - "-iwithprefix", - "-iwithprefixbefore", - "-o", - "-target", - "-x" -}; +// Blacklisted flags which are always removed from the command line. static const char *kBlacklist[] = { - "--param", - "-M", - "-MD", - "-MF", - "-MG", - "-MM", - "-MMD", - "-MP", - "-MQ", - "-MT", - "-Og", - "-Wa,--32", - "-Wa,--64", - "-Wl,--incremental-full", - "-Wl,--incremental-patch,1", - "-Wl,--no-incremental", - "-fbuild-session-file=", - "-fbuild-session-timestamp=", - "-fembed-bitcode", - "-fembed-bitcode-marker", - "-fmodules-validate-once-per-build-session", - "-fno-delete-null-pointer-checks", - "-fno-use-linker-plugin" - "-fno-var-tracking", - "-fno-var-tracking-assignments", - "-fno-enforce-eh-specs", - "-fvar-tracking", - "-fvar-tracking-assignments", - "-fvar-tracking-assignments-toggle", - "-gcc-toolchain", - "-march=", - "-masm=", - "-mcpu=", - "-mfpmath=", - "-mtune=", - "-s", - - //"-B", - //"-f", - //"-pipe", - //"-W", - // TODO - "-Wno-unused-lambda-capture", - "/", - "..", + "-stdlib=libc++" }; -Project::Entry GetCompilationEntryFromCompileCommandEntry(const CompileCommandsEntry& entry) { +// Arguments which are followed by a potentially relative path. We need to make +// all relative paths absolute, otherwise libclang will not resolve them. +const char* kPathArgs[] = { + "-isystem", + "-I", + "-iquote", + "--sysroot=" +}; + +Project::Entry GetCompilationEntryFromCompileCommandEntry(const std::vector& extra_flags, const CompileCommandsEntry& entry) { Project::Entry result; result.filename = NormalizePath(entry.file); - size_t num_args = entry.args.size(); - result.args.reserve(num_args); - for (size_t j = 0; j < num_args; ++j) { - std::string arg = entry.args[j]; + bool make_next_flag_absolute = false; - - bool bad = false; - for (auto& entry : kValueArgs) { - if (StartsWith(arg, entry)) { - bad = true; - continue; - } - } - if (bad) { - ++j; + result.args.reserve(entry.args.size() + extra_flags.size()); + for (size_t i = 0; i < entry.args.size(); ++i) { + std::string arg = entry.args[i]; + + // If blacklist skip. + if (std::any_of(std::begin(kBlacklist), std::end(kBlacklist), [&arg](const char* value) { + return StartsWith(arg, value); + })) { continue; } + // Cleanup path for previous argument. + if (make_next_flag_absolute) { + if (arg.size() > 0 && arg[0] != '/') + arg = NormalizePath(entry.directory + arg); + make_next_flag_absolute = false; + } - for (auto& entry : kBlacklist) { - if (StartsWith(arg, entry)) { - bad = true; - continue; + // Update arg if it is a path. + for (const char* flag_type : kPathArgs) { + if (arg == flag_type) { + make_next_flag_absolute = true; + break; + } + + if (StartsWith(arg, flag_type)) { + std::string path = arg.substr(strlen(flag_type)); + if (path.size() > 0 && path[0] != '/') { + path = NormalizePath(entry.directory + "/" + path); + arg = flag_type + path; + } + break; } } - if (bad) { + if (make_next_flag_absolute) continue; - } - - - if (StartsWith(arg, "-I")) { - std::string path = entry.directory + "/" + arg.substr(2); - path = NormalizePath(path); - arg = "-I" + path; - } result.args.push_back(arg); - //if (StartsWith(arg, "-I") || StartsWith(arg, "-D") || StartsWith(arg, "-std")) } - // TODO/fixme - result.args.push_back("-xc++"); - result.args.push_back("-std=c++11"); + // We don't do any special processing on user-given extra flags. + for (const auto& flag : extra_flags) + result.args.push_back(flag); + + // Clang does not have good hueristics for determining source language. We + // default to C++11 if the user has not specified. + if (!StartsWithAny(entry.args, "-x")) + result.args.push_back("-xc++"); + if (!StartsWithAny(entry.args, "-std=")) + result.args.push_back("-std=c++11"); return result; } -std::vector LoadFromCompileCommandsJson(const std::string& project_directory) { +std::vector LoadFromCompileCommandsJson(const std::vector& extra_flags, const std::string& project_directory) { + // TODO: Fix this function, it may be way faster than libclang's implementation. + optional compile_commands_content = ReadContent(project_directory + "/compile_commands.json"); if (!compile_commands_content) return {}; @@ -168,11 +113,11 @@ std::vector LoadFromCompileCommandsJson(const std::string& proje std::vector result; result.reserve(entries.size()); for (const auto& entry : entries) - result.push_back(GetCompilationEntryFromCompileCommandEntry(entry)); + result.push_back(GetCompilationEntryFromCompileCommandEntry(extra_flags, entry)); return result; } -std::vector LoadFromDirectoryListing(const std::string& project_directory) { +std::vector LoadFromDirectoryListing(const std::vector& extra_flags, const std::string& project_directory) { std::vector result; std::vector args; @@ -185,6 +130,10 @@ std::vector LoadFromDirectoryListing(const std::string& project_ std::cerr << line; args.push_back(line); } + for (const std::string& flag : extra_flags) { + std::cerr << flag << std::endl; + args.push_back(flag); + } std::cerr << std::endl; @@ -201,7 +150,7 @@ std::vector LoadFromDirectoryListing(const std::string& project_ return result; } -std::vector LoadCompilationEntriesFromDirectory(const std::string& project_directory) { +std::vector LoadCompilationEntriesFromDirectory(const std::vector& extra_flags, const std::string& project_directory) { // TODO: Figure out if this function or the clang one is faster. //return LoadFromCompileCommandsJson(project_directory); @@ -210,7 +159,7 @@ std::vector LoadCompilationEntriesFromDirectory(const std::strin CXCompilationDatabase cx_db = clang_CompilationDatabase_fromDirectory(project_directory.c_str(), &cx_db_load_error); if (cx_db_load_error == CXCompilationDatabase_CanNotLoadDatabase) { std::cerr << "Unable to load compile_commands.json located at \"" << project_directory << "\"; using directory listing instead." << std::endl; - return LoadFromDirectoryListing(project_directory); + return LoadFromDirectoryListing(extra_flags, project_directory); } CXCompileCommands cx_commands = clang_CompilationDatabase_getAllCompileCommands(cx_db); @@ -233,7 +182,7 @@ std::vector LoadCompilationEntriesFromDirectory(const std::strin for (unsigned i = 0; i < num_args; ++i) entry.args.push_back(clang::ToString(clang_CompileCommand_getArg(cx_command, i))); - result.push_back(GetCompilationEntryFromCompileCommandEntry(entry)); + result.push_back(GetCompilationEntryFromCompileCommandEntry(extra_flags, entry)); } clang_CompileCommands_dispose(cx_commands); @@ -243,8 +192,8 @@ std::vector LoadCompilationEntriesFromDirectory(const std::strin } } // namespace -void Project::Load(const std::string& directory) { - entries = LoadCompilationEntriesFromDirectory(directory); +void Project::Load(const std::vector& extra_flags, const std::string& directory) { + entries = LoadCompilationEntriesFromDirectory(extra_flags, directory); absolute_path_to_entry_index_.resize(entries.size()); for (int i = 0; i < entries.size(); ++i) diff --git a/src/project.h b/src/project.h index d2553740..cabee6f8 100644 --- a/src/project.h +++ b/src/project.h @@ -28,7 +28,7 @@ struct Project { // discover all files and args. Otherwise, a recursive directory listing of // all *.cpp, *.cc, *.h, and *.hpp files will be used. clang arguments can be // specified in a clang_args file located inside of |directory|. - void Load(const std::string& directory); + void Load(const std::vector& extra_flags, const std::string& directory); // Lookup the CompilationEntry for |filename|. optional FindCompilationEntryForFile(const std::string& filename); diff --git a/src/utils.cc b/src/utils.cc index 9f704cd2..eae689d0 100644 --- a/src/utils.cc +++ b/src/utils.cc @@ -66,18 +66,6 @@ std::string ReplaceAll(const std::string& source, const std::string& from, const return result; } -std::string StringJoin(const std::vector& values) { - std::string result; - bool first = true; - for (auto& entry : values) { - if (!first) - result += ", "; - first = false; - result += entry; - } - return result; -} - static std::vector GetFilesInFolderHelper(std::string folder, bool recursive, std::string output_prefix) { std::vector result; diff --git a/src/utils.h b/src/utils.h index 640d2452..6e5ab5d2 100644 --- a/src/utils.h +++ b/src/utils.h @@ -18,7 +18,25 @@ bool StartsWith(const std::string& value, const std::string& start); bool EndsWith(const std::string& value, const std::string& ending); std::string ReplaceAll(const std::string& source, const std::string& from, const std::string& to); -std::string StringJoin(const std::vector& values); +template +bool StartsWithAny(const TValues& values, const std::string& start) { + return std::any_of(std::begin(values), std::end(values), [&start](const auto& value) { + return StartsWith(value, start); + }); +} + +template +std::string StringJoin(const TValues& values) { + std::string result; + bool first = true; + for (auto& entry : values) { + if (!first) + result += ", "; + first = false; + result += entry; + } + return result; +} // Finds all files in the given folder. This is recursive. std::vector GetFilesInFolder(std::string folder, bool recursive, bool add_folder_to_path);