mirror of
https://github.com/MaskRay/ccls.git
synced 2025-04-03 15:32:09 +00:00
QueryDatabase::{detailed_names,short_names} -> GetSymbol{DetailedName,ShortName}
This commit is contained in:
parent
2fb85732c0
commit
ce6f1c7294
@ -450,10 +450,11 @@ struct TextDocumentCodeActionHandler
|
|||||||
std::unordered_set<std::string> include_absolute_paths;
|
std::unordered_set<std::string> include_absolute_paths;
|
||||||
|
|
||||||
// Find include candidate strings.
|
// Find include candidate strings.
|
||||||
for (int i = 0; i < db->detailed_names.size(); ++i) {
|
for (int i = 0; i < db->symbols.size(); ++i) {
|
||||||
if (include_absolute_paths.size() > kMaxResults)
|
if (include_absolute_paths.size() > kMaxResults)
|
||||||
break;
|
break;
|
||||||
if (db->detailed_names[i].find(include_query) == std::string::npos)
|
if (db->GetSymbolDetailedName(i).find(include_query) ==
|
||||||
|
std::string::npos)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
optional<QueryFileId> decl_file_id =
|
optional<QueryFileId> decl_file_id =
|
||||||
|
@ -124,8 +124,8 @@ gap_penalty(k+1, j) + score[j] : k < j))
|
|||||||
The first dimension can be suppressed since we do not need a matching scheme,
|
The first dimension can be suppressed since we do not need a matching scheme,
|
||||||
which reduces the space complexity from O(N*M) to O(M)
|
which reduces the space complexity from O(N*M) to O(M)
|
||||||
*/
|
*/
|
||||||
int FuzzyEvaluate(const std::string& pattern,
|
int FuzzyEvaluate(std::string_view pattern,
|
||||||
const std::string& str,
|
std::string_view str,
|
||||||
std::vector<int>& score,
|
std::vector<int>& score,
|
||||||
std::vector<int>& dp) {
|
std::vector<int>& dp) {
|
||||||
bool pfirst = true, // aligning the first character of pattern
|
bool pfirst = true, // aligning the first character of pattern
|
||||||
@ -181,7 +181,7 @@ struct WorkspaceSymbolHandler : BaseMessageHandler<Ipc_WorkspaceSymbol> {
|
|||||||
Out_WorkspaceSymbol out;
|
Out_WorkspaceSymbol out;
|
||||||
out.id = request->id;
|
out.id = request->id;
|
||||||
|
|
||||||
LOG_S(INFO) << "[querydb] Considering " << db->detailed_names.size()
|
LOG_S(INFO) << "[querydb] Considering " << db->symbols.size()
|
||||||
<< " candidates for query " << request->params.query;
|
<< " candidates for query " << request->params.query;
|
||||||
|
|
||||||
std::string query = request->params.query;
|
std::string query = request->params.query;
|
||||||
@ -201,10 +201,11 @@ struct WorkspaceSymbolHandler : BaseMessageHandler<Ipc_WorkspaceSymbol> {
|
|||||||
// detailed_names.
|
// detailed_names.
|
||||||
|
|
||||||
// Find exact substring matches.
|
// Find exact substring matches.
|
||||||
for (int i = 0; i < db->detailed_names.size(); ++i) {
|
for (int i = 0; i < db->symbols.size(); ++i) {
|
||||||
if (db->detailed_names[i].find(query) != std::string::npos) {
|
std::string_view detailed_name = db->GetSymbolDetailedName(i);
|
||||||
|
if (detailed_name.find(query) != std::string::npos) {
|
||||||
// Do not show the same entry twice.
|
// Do not show the same entry twice.
|
||||||
if (!inserted_results.insert(db->detailed_names[i]).second)
|
if (!inserted_results.insert(std::string(detailed_name)).second)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (InsertSymbolIntoResult(db, working_files, db->symbols[i],
|
if (InsertSymbolIntoResult(db, working_files, db->symbols[i],
|
||||||
@ -224,10 +225,10 @@ struct WorkspaceSymbolHandler : BaseMessageHandler<Ipc_WorkspaceSymbol> {
|
|||||||
if (!isspace(c))
|
if (!isspace(c))
|
||||||
query_without_space += c;
|
query_without_space += c;
|
||||||
|
|
||||||
for (int i = 0; i < db->short_names.size(); ++i) {
|
for (int i = 0; i < db->symbols.size(); ++i) {
|
||||||
if (SubsequenceMatch(query_without_space, db->short_names[i])) {
|
if (SubsequenceMatch(query_without_space, db->GetSymbolShortName(i))) {
|
||||||
// Do not show the same entry twice.
|
// Do not show the same entry twice.
|
||||||
if (!inserted_results.insert(db->detailed_names[i]).second)
|
if (!inserted_results.insert(std::string(db->GetSymbolDetailedName(i))).second)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (InsertSymbolIntoResult(db, working_files, db->symbols[i],
|
if (InsertSymbolIntoResult(db, working_files, db->symbols[i],
|
||||||
@ -244,7 +245,7 @@ struct WorkspaceSymbolHandler : BaseMessageHandler<Ipc_WorkspaceSymbol> {
|
|||||||
// Sort results with a fuzzy matching algorithm.
|
// Sort results with a fuzzy matching algorithm.
|
||||||
int longest = 0;
|
int longest = 0;
|
||||||
for (int i : result_indices)
|
for (int i : result_indices)
|
||||||
longest = std::max(longest, int(db->short_names[i].size()));
|
longest = std::max(longest, int(db->GetSymbolShortName(i).size()));
|
||||||
|
|
||||||
std::vector<int> score(longest); // score for each position
|
std::vector<int> score(longest); // score for each position
|
||||||
std::vector<int> dp(
|
std::vector<int> dp(
|
||||||
@ -252,7 +253,7 @@ struct WorkspaceSymbolHandler : BaseMessageHandler<Ipc_WorkspaceSymbol> {
|
|||||||
std::vector<std::pair<int, int>> permutation(result_indices.size());
|
std::vector<std::pair<int, int>> permutation(result_indices.size());
|
||||||
for (int i = 0; i < int(result_indices.size()); i++) {
|
for (int i = 0; i < int(result_indices.size()); i++) {
|
||||||
permutation[i] = {
|
permutation[i] = {
|
||||||
FuzzyEvaluate(query, db->short_names[result_indices[i]], score, dp),
|
FuzzyEvaluate(query, db->GetSymbolShortName(result_indices[i]), score, dp),
|
||||||
i};
|
i};
|
||||||
}
|
}
|
||||||
std::sort(permutation.begin(), permutation.end(),
|
std::sort(permutation.begin(), permutation.end(),
|
||||||
|
97
src/query.cc
97
src/query.cc
@ -790,7 +790,8 @@ void QueryDatabase::RemoveUsrs(SymbolKind usr_kind,
|
|||||||
case SymbolKind::Type: {
|
case SymbolKind::Type: {
|
||||||
for (const Usr& usr : to_remove) {
|
for (const Usr& usr : to_remove) {
|
||||||
QueryType& type = types[usr_to_type[usr].id];
|
QueryType& type = types[usr_to_type[usr].id];
|
||||||
RemoveSymbol(type.detailed_name_idx);
|
if (type.symbol_idx != size_t(-1))
|
||||||
|
symbols[type.symbol_idx].kind = SymbolKind::Invalid;
|
||||||
type.def = nullopt;
|
type.def = nullopt;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -798,7 +799,8 @@ void QueryDatabase::RemoveUsrs(SymbolKind usr_kind,
|
|||||||
case SymbolKind::Func: {
|
case SymbolKind::Func: {
|
||||||
for (const Usr& usr : to_remove) {
|
for (const Usr& usr : to_remove) {
|
||||||
QueryFunc& func = funcs[usr_to_func[usr].id];
|
QueryFunc& func = funcs[usr_to_func[usr].id];
|
||||||
RemoveSymbol(func.detailed_name_idx);
|
if (func.symbol_idx != size_t(-1))
|
||||||
|
symbols[func.symbol_idx].kind = SymbolKind::Invalid;
|
||||||
func.def = nullopt;
|
func.def = nullopt;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -806,7 +808,8 @@ void QueryDatabase::RemoveUsrs(SymbolKind usr_kind,
|
|||||||
case SymbolKind::Var: {
|
case SymbolKind::Var: {
|
||||||
for (const Usr& usr : to_remove) {
|
for (const Usr& usr : to_remove) {
|
||||||
QueryVar& var = vars[usr_to_var[usr].id];
|
QueryVar& var = vars[usr_to_var[usr].id];
|
||||||
RemoveSymbol(var.detailed_name_idx);
|
if (var.symbol_idx != size_t(-1))
|
||||||
|
symbols[var.symbol_idx].kind = SymbolKind::Invalid;
|
||||||
var.def = nullopt;
|
var.def = nullopt;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -868,8 +871,8 @@ void QueryDatabase::ImportOrUpdate(
|
|||||||
QueryFile& existing = files[it->second.id];
|
QueryFile& existing = files[it->second.id];
|
||||||
|
|
||||||
existing.def = def.value;
|
existing.def = def.value;
|
||||||
UpdateDetailedNames(&existing.detailed_name_idx, SymbolKind::File,
|
UpdateSymbols(&existing.symbol_idx, SymbolKind::File,
|
||||||
it->second.id, def.value.path, def.value.path);
|
it->second.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -892,9 +895,8 @@ void QueryDatabase::ImportOrUpdate(
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
existing.def = def.value;
|
existing.def = def.value;
|
||||||
UpdateDetailedNames(&existing.detailed_name_idx, SymbolKind::Type,
|
UpdateSymbols(&existing.symbol_idx, SymbolKind::Type,
|
||||||
it->second.id, std::string(def.value.ShortName()),
|
it->second.id);
|
||||||
def.value.detailed_name);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -917,9 +919,8 @@ void QueryDatabase::ImportOrUpdate(
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
existing.def = def.value;
|
existing.def = def.value;
|
||||||
UpdateDetailedNames(&existing.detailed_name_idx, SymbolKind::Func,
|
UpdateSymbols(&existing.symbol_idx, SymbolKind::Func,
|
||||||
it->second.id, std::string(def.value.ShortName()),
|
it->second.id);
|
||||||
def.value.detailed_name);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -943,34 +944,68 @@ void QueryDatabase::ImportOrUpdate(
|
|||||||
|
|
||||||
existing.def = def.value;
|
existing.def = def.value;
|
||||||
if (!def.value.is_local())
|
if (!def.value.is_local())
|
||||||
UpdateDetailedNames(&existing.detailed_name_idx, SymbolKind::Var,
|
UpdateSymbols(&existing.symbol_idx, SymbolKind::Var,
|
||||||
it->second.id, std::string(def.value.ShortName()),
|
it->second.id);
|
||||||
def.value.detailed_name);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QueryDatabase::UpdateDetailedNames(size_t* qualified_name_index,
|
void QueryDatabase::UpdateSymbols(size_t* symbol_idx,
|
||||||
SymbolKind kind,
|
SymbolKind kind,
|
||||||
size_t symbol_index,
|
size_t idx) {
|
||||||
const std::string& short_name,
|
if (*symbol_idx == -1) {
|
||||||
const std::string& detailed_name) {
|
*symbol_idx = symbols.size();
|
||||||
if (*qualified_name_index == -1) {
|
symbols.push_back(SymbolIdx(kind, idx));
|
||||||
short_names.push_back(short_name);
|
|
||||||
detailed_names.push_back(detailed_name);
|
|
||||||
symbols.push_back(SymbolIdx(kind, symbol_index));
|
|
||||||
*qualified_name_index = detailed_names.size() - 1;
|
|
||||||
} else {
|
|
||||||
short_names[*qualified_name_index] = short_name;
|
|
||||||
detailed_names[*qualified_name_index] = detailed_name;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QueryDatabase::RemoveSymbol(size_t idx) {
|
std::string_view QueryDatabase::GetSymbolDetailedName(size_t symbol_idx) const {
|
||||||
if (idx != size_t(-1)) {
|
size_t idx = symbols[symbol_idx].idx;
|
||||||
symbols[idx].kind = SymbolKind::Invalid;
|
switch (symbols[symbol_idx].kind) {
|
||||||
short_names[idx].clear();
|
default:
|
||||||
detailed_names[idx].clear();
|
break;
|
||||||
|
case SymbolKind::File:
|
||||||
|
if (files[idx].def)
|
||||||
|
return files[idx].def->path;
|
||||||
|
break;
|
||||||
|
case SymbolKind::Func:
|
||||||
|
if (funcs[idx].def)
|
||||||
|
return funcs[idx].def->detailed_name;
|
||||||
|
break;
|
||||||
|
case SymbolKind::Type:
|
||||||
|
if (types[idx].def)
|
||||||
|
return types[idx].def->detailed_name;
|
||||||
|
break;
|
||||||
|
case SymbolKind::Var:
|
||||||
|
if (vars[idx].def)
|
||||||
|
return vars[idx].def->detailed_name;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string_view QueryDatabase::GetSymbolShortName(size_t symbol_idx) const {
|
||||||
|
size_t idx = symbols[symbol_idx].idx;
|
||||||
|
switch (symbols[symbol_idx].kind) {
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
case SymbolKind::File:
|
||||||
|
if (files[idx].def)
|
||||||
|
return files[idx].def->path;
|
||||||
|
break;
|
||||||
|
case SymbolKind::Func:
|
||||||
|
if (funcs[idx].def)
|
||||||
|
return funcs[idx].def->ShortName();
|
||||||
|
break;
|
||||||
|
case SymbolKind::Type:
|
||||||
|
if (types[idx].def)
|
||||||
|
return types[idx].def->ShortName();
|
||||||
|
break;
|
||||||
|
case SymbolKind::Var:
|
||||||
|
if (vars[idx].def)
|
||||||
|
return vars[idx].def->ShortName();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_SUITE("query") {
|
TEST_SUITE("query") {
|
||||||
|
22
src/query.h
22
src/query.h
@ -211,7 +211,7 @@ struct QueryFile {
|
|||||||
using DefUpdate = WithFileContent<Def>;
|
using DefUpdate = WithFileContent<Def>;
|
||||||
|
|
||||||
optional<Def> def;
|
optional<Def> def;
|
||||||
size_t detailed_name_idx = (size_t)-1;
|
size_t symbol_idx = (size_t)-1;
|
||||||
|
|
||||||
explicit QueryFile(const std::string& path) {
|
explicit QueryFile(const std::string& path) {
|
||||||
def = Def();
|
def = Def();
|
||||||
@ -241,7 +241,7 @@ struct QueryType {
|
|||||||
std::vector<QueryTypeId> derived;
|
std::vector<QueryTypeId> derived;
|
||||||
std::vector<QueryVarId> instances;
|
std::vector<QueryVarId> instances;
|
||||||
std::vector<QueryLocation> uses;
|
std::vector<QueryLocation> uses;
|
||||||
size_t detailed_name_idx = (size_t)-1;
|
size_t symbol_idx = (size_t)-1;
|
||||||
|
|
||||||
explicit QueryType(const Usr& usr) : usr(usr) {}
|
explicit QueryType(const Usr& usr) : usr(usr) {}
|
||||||
};
|
};
|
||||||
@ -262,7 +262,7 @@ struct QueryFunc {
|
|||||||
std::vector<QueryLocation> declarations;
|
std::vector<QueryLocation> declarations;
|
||||||
std::vector<QueryFuncId> derived;
|
std::vector<QueryFuncId> derived;
|
||||||
std::vector<QueryFuncRef> callers;
|
std::vector<QueryFuncRef> callers;
|
||||||
size_t detailed_name_idx = (size_t)-1;
|
size_t symbol_idx = (size_t)-1;
|
||||||
|
|
||||||
explicit QueryFunc(const Usr& usr) : usr(usr) {}
|
explicit QueryFunc(const Usr& usr) : usr(usr) {}
|
||||||
};
|
};
|
||||||
@ -278,7 +278,7 @@ struct QueryVar {
|
|||||||
optional<Def> def;
|
optional<Def> def;
|
||||||
std::vector<QueryLocation> declarations;
|
std::vector<QueryLocation> declarations;
|
||||||
std::vector<QueryLocation> uses;
|
std::vector<QueryLocation> uses;
|
||||||
size_t detailed_name_idx = (size_t)-1;
|
size_t symbol_idx = (size_t)-1;
|
||||||
|
|
||||||
explicit QueryVar(const Usr& usr) : usr(usr) {}
|
explicit QueryVar(const Usr& usr) : usr(usr) {}
|
||||||
};
|
};
|
||||||
@ -361,10 +361,7 @@ MAKE_HASHABLE(NormalizedPath, t.path);
|
|||||||
// The query database is heavily optimized for fast queries. It is stored
|
// The query database is heavily optimized for fast queries. It is stored
|
||||||
// in-memory.
|
// in-memory.
|
||||||
struct QueryDatabase {
|
struct QueryDatabase {
|
||||||
// Indicies between lookup vectors are related to symbols, ie, index 5 in
|
// All File/Func/Type/Var symbols.
|
||||||
// |detailed_names| matches index 5 in |symbols|.
|
|
||||||
std::vector<std::string> detailed_names;
|
|
||||||
std::vector<std::string> short_names;
|
|
||||||
std::vector<SymbolIdx> symbols;
|
std::vector<SymbolIdx> symbols;
|
||||||
|
|
||||||
// Raw data storage. Accessible via SymbolIdx instances.
|
// Raw data storage. Accessible via SymbolIdx instances.
|
||||||
@ -387,12 +384,9 @@ struct QueryDatabase {
|
|||||||
void ImportOrUpdate(const std::vector<QueryType::DefUpdate>& updates);
|
void ImportOrUpdate(const std::vector<QueryType::DefUpdate>& updates);
|
||||||
void ImportOrUpdate(const std::vector<QueryFunc::DefUpdate>& updates);
|
void ImportOrUpdate(const std::vector<QueryFunc::DefUpdate>& updates);
|
||||||
void ImportOrUpdate(const std::vector<QueryVar::DefUpdate>& updates);
|
void ImportOrUpdate(const std::vector<QueryVar::DefUpdate>& updates);
|
||||||
void UpdateDetailedNames(size_t* qualified_name_index,
|
void UpdateSymbols(size_t* symbol_idx, SymbolKind kind, size_t idx);
|
||||||
SymbolKind kind,
|
std::string_view GetSymbolDetailedName(size_t symbol_idx) const;
|
||||||
size_t symbol_index,
|
std::string_view GetSymbolShortName(size_t symbol_idx) const;
|
||||||
const std::string& short_name,
|
|
||||||
const std::string& detailed_name);
|
|
||||||
void RemoveSymbol(size_t idx);
|
|
||||||
|
|
||||||
// Query the indexing structure to look up symbol id for given Usr.
|
// Query the indexing structure to look up symbol id for given Usr.
|
||||||
optional<QueryFileId> GetQueryFileIdFromPath(const std::string& path);
|
optional<QueryFileId> GetQueryFileIdFromPath(const std::string& path);
|
||||||
|
Loading…
Reference in New Issue
Block a user