mirror of
https://github.com/MaskRay/ccls.git
synced 2025-04-03 07:22:08 +00:00
cache.hierarchicalPath: store cache files as $directory/a/b/c.cc.blob to work around NAME_MAX limitation. cache.retainInMemory: after this number of loads, keep a copy of file index in memory. If set to 1, it avoids cache corruption if the index file is changed after the initial load, which may happen if several language clients open the same project and share the same cache directory. Also rename cacheDirectory cacheFormat to cache.{directory,format}
204 lines
6.2 KiB
C++
204 lines
6.2 KiB
C++
// Copyright 2017-2018 ccls Authors
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
#include "sema_manager.hh"
|
|
#include "fuzzy_match.hh"
|
|
#include "log.hh"
|
|
#include "message_handler.hh"
|
|
#include "pipeline.hh"
|
|
#include "project.hh"
|
|
#include "query.hh"
|
|
|
|
#include <llvm/ADT/STLExtras.h>
|
|
#include <llvm/ADT/StringRef.h>
|
|
#include <llvm/Support/Path.h>
|
|
|
|
#include <algorithm>
|
|
#include <ctype.h>
|
|
#include <functional>
|
|
#include <limits.h>
|
|
using namespace llvm;
|
|
|
|
namespace ccls {
|
|
REFLECT_STRUCT(SymbolInformation, name, kind, location, containerName);
|
|
|
|
void MessageHandler::workspace_didChangeConfiguration(EmptyParam &) {
|
|
for (const std::string &folder : g_config->workspaceFolders)
|
|
project->Load(folder);
|
|
project->Index(wfiles, RequestId());
|
|
|
|
manager->Clear();
|
|
};
|
|
|
|
void MessageHandler::workspace_didChangeWatchedFiles(
|
|
DidChangeWatchedFilesParam ¶m) {
|
|
for (auto &event : param.changes) {
|
|
std::string path = event.uri.GetPath();
|
|
if ((g_config->cache.directory.size() &&
|
|
StringRef(path).startswith(g_config->cache.directory)) ||
|
|
lookupExtension(path).first == LanguageId::Unknown)
|
|
return;
|
|
for (std::string cur = path; cur.size(); cur = sys::path::parent_path(cur))
|
|
if (cur[0] == '.')
|
|
return;
|
|
|
|
IndexMode mode =
|
|
wfiles->GetFile(path) ? IndexMode::Normal : IndexMode::NonInteractive;
|
|
switch (event.type) {
|
|
case FileChangeType::Created:
|
|
case FileChangeType::Changed: {
|
|
pipeline::Index(path, {}, mode, true);
|
|
if (event.type == FileChangeType::Changed) {
|
|
if (mode == IndexMode::Normal)
|
|
manager->OnSave(path);
|
|
else
|
|
manager->OnClose(path);
|
|
}
|
|
break;
|
|
}
|
|
case FileChangeType::Deleted:
|
|
pipeline::Index(path, {}, mode, false);
|
|
manager->OnClose(path);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageHandler::workspace_didChangeWorkspaceFolders(
|
|
DidChangeWorkspaceFoldersParam ¶m) {
|
|
for (const WorkspaceFolder &wf : param.event.removed) {
|
|
std::string root = wf.uri.GetPath();
|
|
EnsureEndsInSlash(root);
|
|
LOG_S(INFO) << "delete workspace folder " << wf.name << ": " << root;
|
|
auto it = llvm::find(g_config->workspaceFolders, root);
|
|
if (it != g_config->workspaceFolders.end()) {
|
|
g_config->workspaceFolders.erase(it);
|
|
{
|
|
// auto &folder = project->root2folder[path];
|
|
// FIXME delete
|
|
}
|
|
project->root2folder.erase(root);
|
|
}
|
|
}
|
|
for (const WorkspaceFolder &wf : param.event.added) {
|
|
std::string root = wf.uri.GetPath();
|
|
EnsureEndsInSlash(root);
|
|
LOG_S(INFO) << "add workspace folder " << wf.name << ": " << root;
|
|
g_config->workspaceFolders.push_back(root);
|
|
project->Load(root);
|
|
}
|
|
|
|
project->Index(wfiles, RequestId());
|
|
|
|
manager->Clear();
|
|
}
|
|
|
|
namespace {
|
|
// Lookup |symbol| in |db| and insert the value into |result|.
|
|
bool AddSymbol(
|
|
DB *db, WorkingFiles *wfiles, const std::vector<uint8_t> &file_set,
|
|
SymbolIdx sym, bool use_detailed,
|
|
std::vector<std::tuple<SymbolInformation, int, SymbolIdx>> *result) {
|
|
std::optional<SymbolInformation> info = GetSymbolInfo(db, sym, true);
|
|
if (!info)
|
|
return false;
|
|
|
|
Maybe<DeclRef> dr;
|
|
bool in_folder = false;
|
|
WithEntity(db, sym, [&](const auto &entity) {
|
|
for (auto &def : entity.def)
|
|
if (def.spell) {
|
|
dr = def.spell;
|
|
if (!in_folder && (in_folder = file_set[def.spell->file_id]))
|
|
break;
|
|
}
|
|
});
|
|
if (!dr) {
|
|
auto &decls = GetNonDefDeclarations(db, sym);
|
|
for (auto &dr1 : decls) {
|
|
dr = dr1;
|
|
if (!in_folder && (in_folder = file_set[dr1.file_id]))
|
|
break;
|
|
}
|
|
}
|
|
if (!in_folder)
|
|
return false;
|
|
|
|
std::optional<Location> ls_location = GetLsLocation(db, wfiles, *dr);
|
|
if (!ls_location)
|
|
return false;
|
|
info->location = *ls_location;
|
|
result->emplace_back(*info, int(use_detailed), sym);
|
|
return true;
|
|
}
|
|
} // namespace
|
|
|
|
void MessageHandler::workspace_symbol(WorkspaceSymbolParam ¶m,
|
|
ReplyOnce &reply) {
|
|
std::vector<SymbolInformation> result;
|
|
const std::string &query = param.query;
|
|
for (auto &folder : param.folders)
|
|
EnsureEndsInSlash(folder);
|
|
std::vector<uint8_t> file_set = db->GetFileSet(param.folders);
|
|
|
|
// {symbol info, matching detailed_name or short_name, index}
|
|
std::vector<std::tuple<SymbolInformation, int, SymbolIdx>> cands;
|
|
bool sensitive = g_config->workspaceSymbol.caseSensitivity;
|
|
|
|
// Find subsequence matches.
|
|
std::string query_without_space;
|
|
query_without_space.reserve(query.size());
|
|
for (char c : query)
|
|
if (!isspace(c))
|
|
query_without_space += c;
|
|
|
|
auto Add = [&](SymbolIdx sym) {
|
|
std::string_view detailed_name = db->GetSymbolName(sym, true);
|
|
int pos = ReverseSubseqMatch(query_without_space, detailed_name, sensitive);
|
|
return pos >= 0 &&
|
|
AddSymbol(db, wfiles, file_set, sym,
|
|
detailed_name.find(':', pos) != std::string::npos,
|
|
&cands) &&
|
|
cands.size() >= g_config->workspaceSymbol.maxNum;
|
|
};
|
|
for (auto &func : db->funcs)
|
|
if (Add({func.usr, Kind::Func}))
|
|
goto done_add;
|
|
for (auto &type : db->types)
|
|
if (Add({type.usr, Kind::Type}))
|
|
goto done_add;
|
|
for (auto &var : db->vars)
|
|
if (var.def.size() && !var.def[0].is_local() && Add({var.usr, Kind::Var}))
|
|
goto done_add;
|
|
done_add:
|
|
|
|
if (g_config->workspaceSymbol.sort && query.size() <= FuzzyMatcher::kMaxPat) {
|
|
// Sort results with a fuzzy matching algorithm.
|
|
int longest = 0;
|
|
for (auto &cand : cands)
|
|
longest = std::max(
|
|
longest, int(db->GetSymbolName(std::get<2>(cand), true).size()));
|
|
FuzzyMatcher fuzzy(query, g_config->workspaceSymbol.caseSensitivity);
|
|
for (auto &cand : cands)
|
|
std::get<1>(cand) = fuzzy.Match(
|
|
db->GetSymbolName(std::get<2>(cand), std::get<1>(cand)), false);
|
|
std::sort(cands.begin(), cands.end(), [](const auto &l, const auto &r) {
|
|
return std::get<1>(l) > std::get<1>(r);
|
|
});
|
|
result.reserve(cands.size());
|
|
for (auto &cand : cands) {
|
|
// Discard awful candidates.
|
|
if (std::get<1>(cand) <= FuzzyMatcher::kMinScore)
|
|
break;
|
|
result.push_back(std::get<0>(cand));
|
|
}
|
|
} else {
|
|
result.reserve(cands.size());
|
|
for (auto &cand : cands)
|
|
result.push_back(std::get<0>(cand));
|
|
}
|
|
|
|
reply(result);
|
|
}
|
|
} // namespace ccls
|