Add clang.pathMappings to reuse cache files with differect source paths

This commit is contained in:
Fangrui Song 2018-09-16 11:07:15 -07:00
parent 34c1ebcefd
commit a18977b9fc
8 changed files with 100 additions and 39 deletions

View File

@ -16,6 +16,7 @@ limitations under the License.
#include "clang_tu.h" #include "clang_tu.h"
#include "clang_utils.h" #include "clang_utils.h"
#include "config.h"
#include <clang/Lex/Lexer.h> #include <clang/Lex/Lexer.h>
using namespace clang; using namespace clang;
@ -67,9 +68,11 @@ Range FromTokenRange(const SourceManager &SM, const LangOptions &LangOpts,
std::unique_ptr<CompilerInvocation> std::unique_ptr<CompilerInvocation>
BuildCompilerInvocation(const std::vector<std::string> &args, BuildCompilerInvocation(const std::vector<std::string> &args,
IntrusiveRefCntPtr<vfs::FileSystem> VFS) { IntrusiveRefCntPtr<vfs::FileSystem> VFS) {
std::string save = "-resource-dir=" + g_config->clang.resourceDir;
std::vector<const char *> cargs; std::vector<const char *> cargs;
for (auto &arg : args) for (auto &arg : args)
cargs.push_back(arg.c_str()); cargs.push_back(arg.c_str());
cargs.push_back(save.c_str());
IntrusiveRefCntPtr<DiagnosticsEngine> Diags( IntrusiveRefCntPtr<DiagnosticsEngine> Diags(
CompilerInstance::createDiagnostics(new DiagnosticOptions)); CompilerInstance::createDiagnostics(new DiagnosticOptions));
std::unique_ptr<CompilerInvocation> CI = std::unique_ptr<CompilerInvocation> CI =

View File

@ -17,3 +17,16 @@ limitations under the License.
Config *g_config; Config *g_config;
thread_local int g_thread_id; thread_local int g_thread_id;
namespace ccls {
void DoPathMapping(std::string &arg) {
for (const std::string &mapping : g_config->clang.pathMappings) {
auto colon = mapping.find(':');
if (colon != std::string::npos) {
auto p = arg.find(mapping.substr(0, colon));
if (p != std::string::npos)
arg.replace(p, colon, mapping.substr(colon + 1));
}
}
}
}

View File

@ -61,6 +61,18 @@ struct Config {
// Additional arguments to pass to clang. // Additional arguments to pass to clang.
std::vector<std::string> extraArgs; std::vector<std::string> extraArgs;
// Translate absolute paths in compile_commands.json entries, .ccls options
// and cache files. This allows to reuse cache files built otherwhere if the
// source paths are different.
//
// This is a list of colon-separated strings, e.g. ["/container:/host"]
//
// An entry of "clang -I /container/include /container/a.cc" will be
// translated to "clang -I /host/include /host/a.cc". This is simple string
// replacement, so "clang /prefix/container/a.cc" will become "clang
// /prefix/host/a.cc".
std::vector<std::string> pathMappings;
// Value to use for clang -resource-dir if not specified. // Value to use for clang -resource-dir if not specified.
// //
// This option defaults to clang -print-resource-dir and should not be // This option defaults to clang -print-resource-dir and should not be
@ -235,7 +247,8 @@ struct Config {
int maxNum = 2000; int maxNum = 2000;
} xref; } xref;
}; };
MAKE_REFLECT_STRUCT(Config::Clang, excludeArgs, extraArgs, resourceDir); MAKE_REFLECT_STRUCT(Config::Clang, excludeArgs, extraArgs, pathMappings,
resourceDir);
MAKE_REFLECT_STRUCT(Config::ClientCapability, snippetSupport); MAKE_REFLECT_STRUCT(Config::ClientCapability, snippetSupport);
MAKE_REFLECT_STRUCT(Config::CodeLens, localVariables); MAKE_REFLECT_STRUCT(Config::CodeLens, localVariables);
MAKE_REFLECT_STRUCT(Config::Completion, caseSensitivity, detailedLabel, MAKE_REFLECT_STRUCT(Config::Completion, caseSensitivity, detailedLabel,
@ -258,3 +271,7 @@ MAKE_REFLECT_STRUCT(Config, compilationDatabaseCommand,
extern Config *g_config; extern Config *g_config;
thread_local extern int g_thread_id; thread_local extern int g_thread_id;
namespace ccls {
void DoPathMapping(std::string &arg);
}

View File

@ -453,6 +453,7 @@ struct Handler_Initialize : BaseMessageHandler<In_InitializeRequest> {
// Ensure there is a resource directory. // Ensure there is a resource directory.
if (g_config->clang.resourceDir.empty()) if (g_config->clang.resourceDir.empty())
g_config->clang.resourceDir = GetDefaultResourceDirectory(); g_config->clang.resourceDir = GetDefaultResourceDirectory();
DoPathMapping(g_config->clang.resourceDir);
LOG_S(INFO) << "Using -resource-dir=" << g_config->clang.resourceDir; LOG_S(INFO) << "Using -resource-dir=" << g_config->clang.resourceDir;
// Send initialization before starting indexers, so we don't send a // Send initialization before starting indexers, so we don't send a
@ -471,10 +472,10 @@ struct Handler_Initialize : BaseMessageHandler<In_InitializeRequest> {
if (g_config->cacheDirectory.size()) { if (g_config->cacheDirectory.size()) {
// Create two cache directories for files inside and outside of the // Create two cache directories for files inside and outside of the
// project. // project.
sys::fs::create_directories(g_config->cacheDirectory + auto len = g_config->projectRoot.size();
EscapeFileName(g_config->projectRoot)); std::string escaped = EscapeFileName(g_config->projectRoot.substr(0, len - 1));
sys::fs::create_directories(g_config->cacheDirectory + '@' + sys::fs::create_directories(g_config->cacheDirectory + escaped);
EscapeFileName(g_config->projectRoot)); sys::fs::create_directories(g_config->cacheDirectory + '@' + escaped);
} }
diag_pub->Init(); diag_pub->Init();

View File

@ -136,12 +136,13 @@ std::string AppendSerializationFormat(const std::string &base) {
std::string GetCachePath(const std::string &source_file) { std::string GetCachePath(const std::string &source_file) {
std::string cache_file; std::string cache_file;
size_t len = g_config->projectRoot.size(); auto len = g_config->projectRoot.size();
if (StartsWith(source_file, g_config->projectRoot)) { if (StartsWith(source_file, g_config->projectRoot)) {
cache_file = EscapeFileName(g_config->projectRoot) + cache_file = EscapeFileName(g_config->projectRoot.substr(0, len - 1)) + '/' +
EscapeFileName(source_file.substr(len)); EscapeFileName(source_file.substr(len));
} else { } else {
cache_file = '@' + EscapeFileName(g_config->projectRoot) + cache_file = '@' +
EscapeFileName(g_config->projectRoot.substr(0, len - 1)) + '/' +
EscapeFileName(source_file); EscapeFileName(source_file);
} }

View File

@ -117,17 +117,6 @@ struct ProjectProcessor {
} }
hash_combine(hash, std::hash<std::string>{}(arg)); hash_combine(hash, std::hash<std::string>{}(arg));
} }
for (size_t i = 1; i < args.size(); i++)
// This is most likely the file path we will be passing to clang. The
// path needs to be absolute, otherwise clang_codeCompleteAt is extremely
// slow. See
// https://github.com/cquery-project/cquery/commit/af63df09d57d765ce12d40007bf56302a0446678.
if (args[i][0] != '-' && EndsWith(args[i], base_name)) {
args[i] = ResolveIfRelative(entry.directory, args[i]);
continue;
}
args.push_back("-resource-dir=" + g_config->clang.resourceDir);
args.push_back("-working-directory=" + entry.directory); args.push_back("-working-directory=" + entry.directory);
if (!command_set.insert(hash).second) { if (!command_set.insert(hash).second) {
@ -197,8 +186,10 @@ ReadCompilerArgumentsFromFile(const std::string &path) {
if (!MBOrErr) if (!MBOrErr)
return {}; return {};
std::vector<std::string> args; std::vector<std::string> args;
for (line_iterator I(*MBOrErr.get(), true, '#'), E; I != E; ++I) for (line_iterator I(*MBOrErr.get(), true, '#'), E; I != E; ++I) {
args.push_back(*I); args.push_back(*I);
DoPathMapping(args.back());
}
return args; return args;
} }
@ -329,9 +320,13 @@ LoadEntriesFromDirectory(ProjectConfig *project,
for (tooling::CompileCommand &Cmd : CDB->getAllCompileCommands()) { for (tooling::CompileCommand &Cmd : CDB->getAllCompileCommands()) {
Project::Entry entry; Project::Entry entry;
entry.directory = NormalizePath(Cmd.Directory); entry.directory = NormalizePath(Cmd.Directory);
DoPathMapping(entry.directory);
entry.filename = entry.filename =
NormalizePath(ResolveIfRelative(entry.directory, Cmd.Filename)); NormalizePath(ResolveIfRelative(entry.directory, Cmd.Filename));
DoPathMapping(entry.filename);
entry.args = std::move(Cmd.CommandLine); entry.args = std::move(Cmd.CommandLine);
for (std::string &arg : entry.args)
DoPathMapping(arg);
proc.Process(entry); proc.Process(entry);
if (Seen.insert(entry.filename).second) if (Seen.insert(entry.filename).second)
result.push_back(entry); result.push_back(entry);

View File

@ -159,26 +159,39 @@ void Reflect(Writer &visitor, std::unordered_map<Usr, V> &map) {
visitor.EndArray(); visitor.EndArray();
} }
// Used by IndexFile::dependencies. Timestamps are emitted for Binary. // Used by IndexFile::dependencies.
void Reflect(Reader &visitor, StringMap<int64_t> &map) { void Reflect(Reader &vis, StringMap<int64_t> &v) {
visitor.IterArray([&](Reader &entry) { std::string name;
std::string name; if (vis.Format() == SerializeFormat::Json) {
Reflect(entry, name); auto &vis1 = static_cast<JsonReader&>(vis);
if (visitor.Format() == SerializeFormat::Binary) for (auto it = vis1.m().MemberBegin(); it != vis1.m().MemberEnd(); ++it)
Reflect(entry, map[name]); v[it->name.GetString()] = it->value.GetInt64();
else } else {
map[name] = 0; vis.IterArray([&](Reader &entry) {
}); Reflect(entry, name);
} Reflect(entry, v[name]);
void Reflect(Writer &visitor, StringMap<int64_t> &map) { });
visitor.StartArray(map.size()); }
for (auto &it : map) { }
std::string key = it.first(); void Reflect(Writer &vis, StringMap<int64_t> &v) {
Reflect(visitor, key); if (vis.Format() == SerializeFormat::Json) {
if (visitor.Format() == SerializeFormat::Binary) auto &vis1 = static_cast<JsonWriter&>(vis);
Reflect(visitor, it.second); vis.StartObject();
for (auto &it : v) {
std::string key = it.first();
vis1.m().Key(key.c_str());
vis1.m().Int64(it.second);
}
vis.EndObject();
} else {
vis.StartArray(v.size());
for (auto &it : v) {
std::string key = it.first();
Reflect(vis, key);
Reflect(vis, it.second);
}
vis.EndArray();
} }
visitor.EndArray();
} }
// TODO: Move this to indexer.cc // TODO: Move this to indexer.cc
@ -448,6 +461,22 @@ Deserialize(SerializeFormat format, const std::string &path,
// Restore non-serialized state. // Restore non-serialized state.
file->path = path; file->path = path;
if (g_config->clang.pathMappings.size()) {
DoPathMapping(file->import_file);
for (std::string &arg : file->args)
DoPathMapping(arg);
for (auto &[_, path] : file->lid2path)
DoPathMapping(path);
for (auto &include : file->includes)
DoPathMapping(include.resolved_path);
StringMap<int64_t> dependencies;
for (auto &it : file->dependencies) {
std::string path = it.first().str();
DoPathMapping(path);
dependencies[path] = it.second;
}
file->dependencies = std::move(dependencies);
}
return file; return file;
} }
} // namespace ccls } // namespace ccls

View File

@ -27,6 +27,7 @@ class JsonReader : public Reader {
public: public:
JsonReader(rapidjson::GenericValue<rapidjson::UTF8<>> *m) : m_(m) {} JsonReader(rapidjson::GenericValue<rapidjson::UTF8<>> *m) : m_(m) {}
SerializeFormat Format() const override { return SerializeFormat::Json; } SerializeFormat Format() const override { return SerializeFormat::Json; }
rapidjson::GenericValue<rapidjson::UTF8<>> &m() { return *m_; }
bool IsBool() override { return m_->IsBool(); } bool IsBool() override { return m_->IsBool(); }
bool IsNull() override { return m_->IsNull(); } bool IsNull() override { return m_->IsNull(); }
@ -95,6 +96,7 @@ class JsonWriter : public Writer {
public: public:
JsonWriter(rapidjson::Writer<rapidjson::StringBuffer> *m) : m_(m) {} JsonWriter(rapidjson::Writer<rapidjson::StringBuffer> *m) : m_(m) {}
SerializeFormat Format() const override { return SerializeFormat::Json; } SerializeFormat Format() const override { return SerializeFormat::Json; }
rapidjson::Writer<rapidjson::StringBuffer> &m() { return *m_; }
void Null() override { m_->Null(); } void Null() override { m_->Null(); }
void Bool(bool x) override { m_->Bool(x); } void Bool(bool x) override { m_->Bool(x); }