mirror of
https://github.com/MaskRay/ccls.git
synced 2025-01-19 03:55:49 +00:00
Add loose mode in project.cc
This commit is contained in:
parent
f9d7361953
commit
e83fce65c2
@ -289,8 +289,13 @@ void TraceMe() {
|
|||||||
std::string GetExternalCommandOutput(const std::vector<std::string>& command,
|
std::string GetExternalCommandOutput(const std::vector<std::string>& command,
|
||||||
std::string_view input) {
|
std::string_view input) {
|
||||||
int pin[2], pout[2];
|
int pin[2], pout[2];
|
||||||
pipe(pin);
|
if (pipe(pin) < 0)
|
||||||
pipe(pout);
|
return "";
|
||||||
|
if (pipe(pout) < 0) {
|
||||||
|
close(pin[0]);
|
||||||
|
close(pin[1]);
|
||||||
|
return "";
|
||||||
|
}
|
||||||
pid_t child = fork();
|
pid_t child = fork();
|
||||||
if (child == 0) {
|
if (child == 0) {
|
||||||
dup2(pout[0], 0);
|
dup2(pout[0], 0);
|
||||||
|
118
src/project.cc
118
src/project.cc
@ -97,15 +97,18 @@ bool ShouldAddToAngleIncludes(const std::string& arg) {
|
|||||||
return StartsWithAny(arg, kAngleIncludeArgs);
|
return StartsWithAny(arg, kAngleIncludeArgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
optional<std::string> SourceFileType(const std::string& path) {
|
// FIXME
|
||||||
|
enum class LanguageId { Unknown = 0, C = 1, Cpp = 2, ObjC = 3, ObjCpp = 4 };
|
||||||
|
|
||||||
|
optional<LanguageId> SourceFileType(const std::string& path) {
|
||||||
if (EndsWith(path, ".c"))
|
if (EndsWith(path, ".c"))
|
||||||
return std::string("c");
|
return LanguageId::C;
|
||||||
else if (EndsWith(path, ".cpp") || EndsWith(path, ".cc"))
|
else if (EndsWith(path, ".cpp") || EndsWith(path, ".cc"))
|
||||||
return std::string("c++");
|
return LanguageId::Cpp;
|
||||||
else if (EndsWith(path, ".mm"))
|
else if (EndsWith(path, ".mm"))
|
||||||
return std::string("objective-c++");
|
return LanguageId::ObjCpp;
|
||||||
else if (EndsWith(path, ".m"))
|
else if (EndsWith(path, ".m"))
|
||||||
return std::string("objective-c");
|
return LanguageId::ObjC;
|
||||||
return nullopt;
|
return nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -131,36 +134,39 @@ Project::Entry GetCompilationEntryFromCompileCommandEntry(
|
|||||||
result.filename = NormalizePathWithTestOptOut(entry.file);
|
result.filename = NormalizePathWithTestOptOut(entry.file);
|
||||||
std::string base_name = GetBaseName(entry.file);
|
std::string base_name = GetBaseName(entry.file);
|
||||||
|
|
||||||
|
// If |compilationDatabaseCommand| is specified, the external command provides
|
||||||
|
// us the JSON compilation database which should be strict. We should do very
|
||||||
|
// little processing on |entry.args|. The
|
||||||
|
bool loose = init_opts->compilationDatabaseCommand.empty();
|
||||||
|
|
||||||
size_t i = 0;
|
size_t i = 0;
|
||||||
|
|
||||||
// Strip all arguments consisting the compiler command,
|
if (loose) {
|
||||||
// as there may be non-compiler related commands beforehand,
|
// Strip all arguments consisting the compiler command,
|
||||||
// ie, compiler schedular such as goma. This allows correct parsing for
|
// as there may be non-compiler related commands beforehand,
|
||||||
// command lines like "goma clang -c foo".
|
// ie, compiler schedular such as goma. This allows correct parsing for
|
||||||
std::string::size_type dot;
|
// command lines like "goma clang -c foo".
|
||||||
while (i < entry.args.size() && entry.args[i][0] != '-' &&
|
std::string::size_type dot;
|
||||||
// Do not skip over main source filename
|
while (i < entry.args.size() && entry.args[i][0] != '-' &&
|
||||||
NormalizePathWithTestOptOut(entry.args[i]) != result.filename &&
|
// Do not skip over main source filename
|
||||||
// There may be other filenames (e.g. more than one source filenames)
|
NormalizePathWithTestOptOut(entry.args[i]) != result.filename &&
|
||||||
// preceding main source filename. We use a heuristic here. `.` may
|
// There may be other filenames (e.g. more than one source filenames)
|
||||||
// occur in both command names and source filenames. If `.` occurs in
|
// preceding main source filename. We use a heuristic here. `.` may
|
||||||
// the last 4 bytes of entry.args[i] and not followed by a digit, e.g.
|
// occur in both command names and source filenames. If `.` occurs in
|
||||||
// .c .cpp, We take it as a source filename. Others (like ./a/b/goma
|
// the last 4 bytes of entry.args[i] and not followed by a digit, e.g.
|
||||||
// clang-4.0) are seen as commands.
|
// .c .cpp, We take it as a source filename. Others (like ./a/b/goma
|
||||||
((dot = entry.args[i].rfind('.')) == std::string::npos ||
|
// clang-4.0) are seen as commands.
|
||||||
dot + 4 < entry.args[i].size() || isdigit(entry.args[i][dot + 1]) ||
|
((dot = entry.args[i].rfind('.')) == std::string::npos ||
|
||||||
!entry.args[i].compare(dot + 1, 3, "exe")))
|
dot + 4 < entry.args[i].size() || isdigit(entry.args[i][dot + 1]) ||
|
||||||
++i;
|
!entry.args[i].compare(dot + 1, 3, "exe")))
|
||||||
// Include the compiler in the args.
|
++i;
|
||||||
|
} else {
|
||||||
|
if (entry.args.size())
|
||||||
|
i = 1;
|
||||||
|
}
|
||||||
|
// Compiler driver.
|
||||||
if (i > 0)
|
if (i > 0)
|
||||||
result.args.push_back(entry.args[i - 1]);
|
result.args.push_back(entry.args[i - 1]);
|
||||||
else {
|
|
||||||
// TODO Drop this back compatibility
|
|
||||||
// Args probably came from a /.cquery file, which likely has just flags.
|
|
||||||
// clang_parseTranslationUnit2FullArgv() expects the binary path as the
|
|
||||||
// first arg, so the first flag would end up being ignored. Add a dummy.
|
|
||||||
result.args.push_back("clang++");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add -working-directory if not provided.
|
// Add -working-directory if not provided.
|
||||||
if (!AnyStartsWith(entry.args, "-working-directory")) {
|
if (!AnyStartsWith(entry.args, "-working-directory")) {
|
||||||
@ -168,17 +174,24 @@ Project::Entry GetCompilationEntryFromCompileCommandEntry(
|
|||||||
result.args.push_back(entry.directory);
|
result.args.push_back(entry.directory);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clang does not have good hueristics for determining source language, we
|
if (loose) {
|
||||||
// should explicitly specify it.
|
// FIXME
|
||||||
if (optional<std::string> file_type = SourceFileType(entry.file)) {
|
if (auto lang = SourceFileType(entry.file)) {
|
||||||
if (!AnyStartsWith(entry.args, "-x")) {
|
if (!AnyStartsWith(entry.args, "-x")) {
|
||||||
result.args.push_back("-x" + *file_type);
|
switch (*lang) {
|
||||||
}
|
case LanguageId::Unknown: break;
|
||||||
if (!AnyStartsWith(entry.args, "-std=")) {
|
case LanguageId::C: result.args.push_back("-xc"); break;
|
||||||
if (*file_type == "c")
|
case LanguageId::Cpp: result.args.push_back("-xc++"); break;
|
||||||
result.args.push_back("-std=gnu11");
|
case LanguageId::ObjC: result.args.push_back("-xobjective-c"); break;
|
||||||
else if (*file_type == "c++")
|
case LanguageId::ObjCpp: result.args.push_back("-xobjective-cpp"); break;
|
||||||
result.args.push_back("-std=c++14");
|
}
|
||||||
|
}
|
||||||
|
if (!AnyStartsWith(entry.args, "-std=")) {
|
||||||
|
if (*lang == LanguageId::C)
|
||||||
|
result.args.push_back("-std=gnu11");
|
||||||
|
else if (*lang == LanguageId::Cpp)
|
||||||
|
result.args.push_back("-std=c++14");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -215,9 +228,7 @@ Project::Entry GetCompilationEntryFromCompileCommandEntry(
|
|||||||
next_flag_is_path = false;
|
next_flag_is_path = false;
|
||||||
add_next_flag_to_quote_dirs = false;
|
add_next_flag_to_quote_dirs = false;
|
||||||
add_next_flag_to_angle_dirs = false;
|
add_next_flag_to_angle_dirs = false;
|
||||||
}
|
} else {
|
||||||
|
|
||||||
else {
|
|
||||||
// Check to see if arg is a path and needs to be updated.
|
// Check to see if arg is a path and needs to be updated.
|
||||||
for (const std::string& flag_type : kPathArgs) {
|
for (const std::string& flag_type : kPathArgs) {
|
||||||
// {"-I", "foo"} style.
|
// {"-I", "foo"} style.
|
||||||
@ -551,7 +562,18 @@ Project::Entry Project::FindCompilationEntryForFile(
|
|||||||
result.filename = filename;
|
result.filename = filename;
|
||||||
if (!best_entry) {
|
if (!best_entry) {
|
||||||
// FIXME
|
// FIXME
|
||||||
result.args.push_back("clang++");
|
if (auto lang = SourceFileType(filename)) {
|
||||||
|
switch (*lang) {
|
||||||
|
case LanguageId::C:
|
||||||
|
result.args.push_back("clang");
|
||||||
|
break;
|
||||||
|
case LanguageId::Cpp:
|
||||||
|
result.args.push_back("clang++");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
result.args.push_back(filename);
|
result.args.push_back(filename);
|
||||||
} else {
|
} else {
|
||||||
result.args = best_entry->args;
|
result.args = best_entry->args;
|
||||||
@ -687,9 +709,9 @@ TEST_SUITE("Project") {
|
|||||||
TEST_CASE("Implied binary") {
|
TEST_CASE("Implied binary") {
|
||||||
CheckFlags(
|
CheckFlags(
|
||||||
"/home/user", "/home/user/foo/bar.cc",
|
"/home/user", "/home/user/foo/bar.cc",
|
||||||
/* raw */ {"-DDONT_IGNORE_ME"},
|
/* raw */ {"clang", "-DDONT_IGNORE_ME"},
|
||||||
/* expected */
|
/* expected */
|
||||||
{"clang++", "-working-directory", "/home/user", "-xc++", "-std=c++14",
|
{"clang", "-working-directory", "/home/user", "-xc++", "-std=c++14",
|
||||||
"-DDONT_IGNORE_ME", "-resource-dir=/w/resource_dir/",
|
"-DDONT_IGNORE_ME", "-resource-dir=/w/resource_dir/",
|
||||||
"-Wno-unknown-warning-option", "-fparse-all-comments"});
|
"-Wno-unknown-warning-option", "-fparse-all-comments"});
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user