ccls/src/query_utils.cc

592 lines
17 KiB
C++
Raw Normal View History

2017-06-15 05:32:23 +00:00
#include "query_utils.h"
#include "queue_manager.h"
#include <loguru.hpp>
2017-06-16 17:14:09 +00:00
#include <climits>
2018-02-07 05:26:38 +00:00
#include <queue>
2017-06-16 17:14:09 +00:00
namespace {
// Computes roughly how long |range| is.
int ComputeRangeSize(const Range& range) {
if (range.start.line != range.end.line)
return INT_MAX;
return range.end.column - range.start.column;
}
2018-02-09 05:11:35 +00:00
} // namespace
2018-02-11 04:30:27 +00:00
Maybe<Use> GetDefinitionSpellingOfSymbol(QueryDatabase* db,
QueryFuncId id) {
QueryFunc& func = db->funcs[id.id];
if (func.def)
return func.def->spell;
2017-06-15 05:32:23 +00:00
return nullopt;
}
2018-02-11 04:30:27 +00:00
Maybe<Use> GetDefinitionSpellingOfSymbol(QueryDatabase* db,
SymbolIdx sym) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
2018-02-10 03:07:45 +00:00
QueryType& type = db->GetType(sym);
if (type.def)
2018-02-11 04:30:27 +00:00
return type.def->spell;
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Func: {
2018-02-10 03:07:45 +00:00
QueryFunc& func = db->GetFunc(sym);
if (func.def)
return func.def->spell;
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Var: {
2018-02-10 03:07:45 +00:00
QueryVar& var = db->GetVar(sym);
if (var.def)
2018-02-11 04:30:27 +00:00
return var.def->spell;
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::File:
case SymbolKind::Invalid: {
assert(false && "unexpected");
break;
}
2017-06-15 05:32:23 +00:00
}
return nullopt;
}
Maybe<Use> GetDefinitionExtentOfSymbol(QueryDatabase* db, SymbolIdx sym) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
2018-02-10 03:07:45 +00:00
QueryType& type = db->GetType(sym);
if (type.def)
return type.def->extent;
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Func: {
2018-02-10 03:07:45 +00:00
QueryFunc& func = db->GetFunc(sym);
if (func.def)
return Use(*func.def->extent);
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Var: {
2018-02-10 03:07:45 +00:00
QueryVar& var = db->GetVar(sym);
if (var.def)
return var.def->extent;
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::File:
return Use(Range(Position(0, 0), Position(0, 0)), sym.id, sym.kind,
Role::None);
2017-09-22 01:14:57 +00:00
case SymbolKind::Invalid: {
assert(false && "unexpected");
break;
}
2017-06-15 05:32:23 +00:00
}
return nullopt;
}
Maybe<QueryFileId> GetDeclarationFileForSymbol(QueryDatabase* db,
SymbolIdx sym) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
2018-02-10 03:07:45 +00:00
QueryType& type = db->GetType(sym);
if (type.def && type.def->spell)
return db->GetFileId(*type.def->spell);
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Func: {
2018-02-10 03:07:45 +00:00
QueryFunc& func = db->GetFunc(sym);
if (!func.declarations.empty())
return db->GetFileId(func.declarations[0]);
if (func.def && func.def->spell)
return db->GetFileId(*func.def->spell);
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::Var: {
2018-02-10 03:07:45 +00:00
QueryVar& var = db->GetVar(sym);
if (var.def && var.def->spell)
return db->GetFileId(*var.def->spell);
2017-09-22 01:14:57 +00:00
break;
}
case SymbolKind::File:
return QueryFileId(sym.id);
2017-09-22 01:14:57 +00:00
case SymbolKind::Invalid: {
assert(false && "unexpected");
break;
}
2017-06-15 05:32:23 +00:00
}
return nullopt;
}
std::vector<Use> ToUses(QueryDatabase* db,
const std::vector<QueryFuncId>& ids) {
std::vector<Use> ret;
2018-02-10 06:51:58 +00:00
ret.reserve(ids.size());
for (auto id : ids) {
QueryFunc& func = db->funcs[id.id];
if (func.def && func.def->spell)
ret.push_back(*func.def->spell);
2018-02-10 06:51:58 +00:00
else if (func.declarations.size())
ret.push_back(func.declarations[0]);
}
return ret;
}
std::vector<Use> ToUses(QueryDatabase* db,
const std::vector<QueryTypeId>& ids) {
std::vector<Use> ret;
2018-02-10 06:51:58 +00:00
ret.reserve(ids.size());
for (auto id : ids) {
QueryType& type = db->types[id.id];
if (type.def && type.def->spell)
ret.push_back(*type.def->spell);
2018-02-10 06:51:58 +00:00
}
return ret;
}
std::vector<Use> ToUses(QueryDatabase* db, const std::vector<QueryVarId>& ids) {
std::vector<Use> ret;
2018-02-10 06:51:58 +00:00
ret.reserve(ids.size());
for (auto id : ids) {
QueryVar& var = db->vars[id.id];
if (var.def && var.def->spell)
ret.push_back(*var.def->spell);
2018-02-10 06:51:58 +00:00
else if (var.declarations.size())
ret.push_back(var.declarations[0]);
}
return ret;
}
std::vector<Use> GetUsesOfSymbol(QueryDatabase* db,
SymbolIdx sym,
bool include_decl) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
QueryType& type = db->GetType(sym);
std::vector<Use> ret = type.uses;
if (include_decl && type.def && type.def->spell)
ret.push_back(*type.def->spell);
2018-01-29 07:05:51 +00:00
return ret;
2017-09-22 01:14:57 +00:00
}
case SymbolKind::Func: {
QueryFunc& func = db->GetFunc(sym);
std::vector<Use> ret = func.uses;
if (include_decl) {
if (func.def && func.def->spell)
ret.push_back(*func.def->spell);
AddRange(&ret, func.declarations);
}
2018-02-09 05:11:35 +00:00
return ret;
2017-09-22 01:14:57 +00:00
}
case SymbolKind::Var: {
QueryVar& var = db->GetVar(sym);
std::vector<Use> ret = var.uses;
if (include_decl) {
if (var.def && var.def->spell)
ret.push_back(*var.def->spell);
2018-01-30 00:27:43 +00:00
ret.insert(ret.end(), var.declarations.begin(), var.declarations.end());
}
return ret;
2017-09-22 01:14:57 +00:00
}
case SymbolKind::File:
case SymbolKind::Invalid: {
assert(false && "unexpected");
2018-02-09 05:11:35 +00:00
return {};
2017-09-22 01:14:57 +00:00
}
2017-06-15 05:32:23 +00:00
}
}
std::vector<Use> GetDeclarationsOfSymbolForGotoDefinition(
2017-09-22 01:14:57 +00:00
QueryDatabase* db,
SymbolIdx sym) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
// Returning the definition spelling of a type is a hack (and is why the
// function has the postfix `ForGotoDefintion`, but it lets the user
// jump to the start of a type if clicking goto-definition on the same
// type from within the type definition.
QueryType& type = db->GetType(sym);
if (type.def) {
Maybe<Use> def = type.def->spell;
if (def)
return {*def};
2017-09-22 01:14:57 +00:00
}
break;
}
2018-02-09 05:11:35 +00:00
case SymbolKind::Func:
return db->GetFunc(sym).declarations;
2018-02-09 05:11:35 +00:00
case SymbolKind::Var:
return db->GetVar(sym).declarations;
2017-09-22 01:14:57 +00:00
default:
break;
2017-06-15 05:32:23 +00:00
}
return {};
}
bool HasCallersOnSelfOrBaseOrDerived(QueryDatabase* db, QueryFunc& root) {
// Check self.
if (!root.uses.empty())
return true;
// Check for base calls.
2018-02-04 00:20:14 +00:00
std::queue<QueryFunc*> queue;
EachWithGen<QueryFunc>(db->funcs, root.def->base, [&](QueryFunc& func) {
queue.push(&func);
});
2017-12-19 06:15:46 +00:00
while (!queue.empty()) {
2018-02-04 00:20:14 +00:00
QueryFunc& func = *queue.front();
2017-12-19 06:15:46 +00:00
queue.pop();
if (!func.uses.empty())
return true;
2017-12-19 06:15:46 +00:00
if (func.def)
2018-02-04 00:20:14 +00:00
EachWithGen<QueryFunc>(db->funcs, func.def->base, [&](QueryFunc& func1) {
queue.push(&func1);
});
}
// Check for derived calls.
2018-02-04 00:20:14 +00:00
EachWithGen<QueryFunc>(db->funcs, root.derived, [&](QueryFunc& func1) {
queue.push(&func1);
});
while (!queue.empty()) {
2018-02-04 00:20:14 +00:00
QueryFunc& func = *queue.front();
queue.pop();
if (!func.uses.empty())
return true;
2018-02-04 00:20:14 +00:00
EachWithGen<QueryFunc>(db->funcs, func.derived, [&](QueryFunc& func1) {
queue.push(&func1);
});
}
return false;
}
std::vector<Use> GetCallersForAllBaseFunctions(QueryDatabase* db,
QueryFunc& root) {
std::vector<Use> callers;
2018-01-30 04:18:08 +00:00
if (!root.def)
return callers;
2017-06-15 05:32:23 +00:00
2018-02-04 00:20:14 +00:00
std::queue<QueryFunc*> queue;
EachWithGen<QueryFunc>(db->funcs, root.def->base, [&](QueryFunc& func1) {
queue.push(&func1);
});
2017-12-19 06:15:46 +00:00
while (!queue.empty()) {
2018-02-04 00:20:14 +00:00
QueryFunc& func = *queue.front();
2017-12-19 06:15:46 +00:00
queue.pop();
2017-06-15 05:32:23 +00:00
AddRange(&callers, func.uses);
2017-12-19 06:15:46 +00:00
if (func.def)
2018-02-04 00:20:14 +00:00
EachWithGen<QueryFunc>(db->funcs, func.def->base, [&](QueryFunc& func1) {
queue.push(&func1);
});
2017-06-15 05:32:23 +00:00
}
return callers;
}
std::vector<Use> GetCallersForAllDerivedFunctions(QueryDatabase* db,
QueryFunc& root) {
std::vector<Use> callers;
2017-06-15 05:32:23 +00:00
2018-02-04 00:20:14 +00:00
std::queue<QueryFunc*> queue;
EachWithGen<QueryFunc>(db->funcs, root.derived, [&](QueryFunc& func) {
queue.push(&func);
});
2017-06-15 05:32:23 +00:00
while (!queue.empty()) {
2018-02-04 00:20:14 +00:00
QueryFunc& func = *queue.front();
2017-06-15 05:32:23 +00:00
queue.pop();
2018-02-04 00:20:14 +00:00
EachWithGen<QueryFunc>(db->funcs, func.derived, [&](QueryFunc& func1) {
queue.push(&func1);
});
AddRange(&callers, func.uses);
2017-06-15 05:32:23 +00:00
}
return callers;
}
2017-09-22 01:14:57 +00:00
optional<lsPosition> GetLsPosition(WorkingFile* working_file,
const Position& position) {
2017-06-15 05:32:23 +00:00
if (!working_file)
return lsPosition(position.line, position.column);
2017-06-15 05:32:23 +00:00
int column = position.column;
2018-01-30 00:27:43 +00:00
optional<int> start =
working_file->GetBufferPosFromIndexPos(position.line, &column, false);
2017-06-15 05:32:23 +00:00
if (!start)
return nullopt;
return lsPosition(*start, column);
2017-06-15 05:32:23 +00:00
}
optional<lsRange> GetLsRange(WorkingFile* working_file, const Range& location) {
if (!working_file) {
2018-01-30 00:27:43 +00:00
return lsRange(lsPosition(location.start.line, location.start.column),
lsPosition(location.end.line, location.end.column));
2017-06-15 05:32:23 +00:00
}
int start_column = location.start.column, end_column = location.end.column;
2018-01-30 00:27:43 +00:00
optional<int> start = working_file->GetBufferPosFromIndexPos(
location.start.line, &start_column, false);
optional<int> end = working_file->GetBufferPosFromIndexPos(location.end.line,
&end_column, true);
2017-06-15 05:32:23 +00:00
if (!start || !end)
return nullopt;
// If remapping end fails (end can never be < start), just guess that the
// final location didn't move. This only screws up the highlighted code
// region if we guess wrong, so not a big deal.
//
// Remapping fails often in C++ since there are a lot of "};" at the end of
// class/struct definitions.
if (*end < *start)
*end = *start + (location.end.line - location.start.line);
2018-01-15 06:53:51 +00:00
if (*start == *end && start_column > end_column)
end_column = start_column;
2017-06-15 05:32:23 +00:00
return lsRange(lsPosition(*start, start_column),
lsPosition(*end, end_column));
2017-06-15 05:32:23 +00:00
}
2017-09-22 01:14:57 +00:00
lsDocumentUri GetLsDocumentUri(QueryDatabase* db,
QueryFileId file_id,
std::string* path) {
QueryFile& file = db->files[file_id.id];
if (file.def) {
*path = file.def->path;
2017-06-15 05:32:23 +00:00
return lsDocumentUri::FromPath(*path);
2017-09-22 01:14:57 +00:00
} else {
2017-06-15 05:32:23 +00:00
*path = "";
return lsDocumentUri::FromPath("");
}
}
lsDocumentUri GetLsDocumentUri(QueryDatabase* db, QueryFileId file_id) {
QueryFile& file = db->files[file_id.id];
if (file.def) {
return lsDocumentUri::FromPath(file.def->path);
2017-09-22 01:14:57 +00:00
} else {
2017-06-15 05:32:23 +00:00
return lsDocumentUri::FromPath("");
}
}
2017-09-22 01:14:57 +00:00
optional<lsLocation> GetLsLocation(QueryDatabase* db,
WorkingFiles* working_files,
2018-02-09 05:11:35 +00:00
Reference ref) {
2017-06-15 05:32:23 +00:00
std::string path;
Maybe<QueryFileId> file_id = db->GetFileId(ref);
if (!file_id)
return nullopt;
lsDocumentUri uri = GetLsDocumentUri(db, *file_id, &path);
2017-09-22 01:14:57 +00:00
optional<lsRange> range =
2018-02-09 05:11:35 +00:00
GetLsRange(working_files->GetFileByFilename(path), ref.range);
2017-06-15 05:32:23 +00:00
if (!range)
return nullopt;
return lsLocation(uri, *range);
}
optional<lsLocationEx> GetLsLocationEx(QueryDatabase* db,
WorkingFiles* working_files,
Use use,
bool extension) {
optional<lsLocation> ls_loc = GetLsLocation(db, working_files, use);
if (!ls_loc)
return nullopt;
lsLocationEx ret;
ret.lsLocation::operator=(*ls_loc);
if (extension)
switch (use.kind) {
default:
break;
case SymbolKind::Func: {
QueryFunc& func = db->GetFunc(use);
if (func.def)
ret.containerName = std::string_view(func.def->detailed_name);
break;
}
case SymbolKind::Type: {
QueryType& type = db->GetType(use);
if (type.def)
ret.containerName = std::string_view(type.def->detailed_name);
break;
}
case SymbolKind::Var: {
QueryVar& var = db->GetVar(use);
if (var.def)
ret.containerName = std::string_view(var.def->detailed_name);
break;
}
}
return ret;
}
2017-12-12 05:20:29 +00:00
std::vector<lsLocation> GetLsLocations(
2017-09-22 01:14:57 +00:00
QueryDatabase* db,
WorkingFiles* working_files,
const std::vector<Use>& uses) {
2017-06-15 05:32:23 +00:00
std::unordered_set<lsLocation> unique_locations;
for (Use use : uses) {
2017-09-22 01:14:57 +00:00
optional<lsLocation> location =
GetLsLocation(db, working_files, use);
2017-06-15 05:32:23 +00:00
if (!location)
continue;
unique_locations.insert(*location);
}
2017-12-12 05:20:29 +00:00
std::vector<lsLocation> result;
2017-06-15 05:32:23 +00:00
result.reserve(unique_locations.size());
result.assign(unique_locations.begin(), unique_locations.end());
return result;
}
// Returns a symbol. The symbol will have *NOT* have a location assigned.
2017-09-22 01:14:57 +00:00
optional<lsSymbolInformation> GetSymbolInfo(QueryDatabase* db,
WorkingFiles* working_files,
SymbolIdx sym,
bool use_short_name) {
2018-02-09 17:42:10 +00:00
switch (sym.kind) {
2017-09-22 01:14:57 +00:00
case SymbolKind::File: {
QueryFile& file = db->GetFile(sym);
if (!file.def)
2018-02-09 17:42:10 +00:00
break;
2017-06-15 05:32:23 +00:00
2017-09-22 01:14:57 +00:00
lsSymbolInformation info;
info.name = file.def->path;
2017-09-22 01:14:57 +00:00
info.kind = lsSymbolKind::File;
return info;
2017-06-15 05:32:23 +00:00
}
2017-09-22 01:14:57 +00:00
case SymbolKind::Type: {
2018-02-10 03:07:45 +00:00
QueryType& type = db->GetType(sym);
if (!type.def)
2018-02-09 17:42:10 +00:00
break;
2017-09-22 01:14:57 +00:00
lsSymbolInformation info;
if (use_short_name)
info.name = type.def->ShortName();
else
info.name = type.def->detailed_name;
if (type.def->detailed_name.c_str() != type.def->ShortName())
info.containerName = type.def->detailed_name;
2018-01-24 08:19:42 +00:00
// TODO ClangSymbolKind -> lsSymbolKind
switch (type.def->kind) {
2018-01-30 00:27:43 +00:00
default:
info.kind = lsSymbolKind::Class;
break;
case ClangSymbolKind::Namespace:
info.kind = lsSymbolKind::Namespace;
break;
2018-01-24 08:19:42 +00:00
}
2017-09-22 01:14:57 +00:00
return info;
}
case SymbolKind::Func: {
2018-02-10 03:07:45 +00:00
QueryFunc& func = db->GetFunc(sym);
if (!func.def)
2018-02-09 17:42:10 +00:00
break;
2017-09-22 01:14:57 +00:00
lsSymbolInformation info;
if (use_short_name)
info.name = func.def->ShortName();
else
info.name = func.def->detailed_name;
info.containerName = func.def->detailed_name;
2017-09-22 01:14:57 +00:00
info.kind = lsSymbolKind::Function;
if (func.def->declaring_type) {
QueryType& container = db->types[func.def->declaring_type->id];
if (container.def)
2017-09-22 01:14:57 +00:00
info.kind = lsSymbolKind::Method;
}
return info;
}
case SymbolKind::Var: {
2018-02-10 03:07:45 +00:00
QueryVar& var = db->GetVar(sym);
if (!var.def)
2018-02-09 17:42:10 +00:00
break;
2017-09-22 01:14:57 +00:00
lsSymbolInformation info;
if (use_short_name)
info.name = var.def->ShortName();
else
info.name = var.def->detailed_name;
info.containerName = var.def->detailed_name;
2017-09-22 01:14:57 +00:00
info.kind = lsSymbolKind::Variable;
return info;
}
2018-02-09 17:42:10 +00:00
case SymbolKind::Invalid:
break;
}
2017-06-15 05:32:23 +00:00
return nullopt;
}
// TODO Sort only by range length, not |kind| or |idx|
2017-09-22 01:14:57 +00:00
std::vector<SymbolRef> FindSymbolsAtLocation(WorkingFile* working_file,
QueryFile* file,
lsPosition position) {
2017-06-15 05:32:23 +00:00
std::vector<SymbolRef> symbols;
symbols.reserve(1);
int target_line = position.line;
int target_column = position.character;
2017-06-15 05:32:23 +00:00
if (working_file) {
optional<int> index_line = working_file->GetIndexPosFromBufferPos(
target_line, &target_column, false);
2017-06-15 05:32:23 +00:00
if (index_line)
target_line = *index_line;
}
2018-02-09 17:42:10 +00:00
for (const SymbolRef& sym : file->def->all_symbols) {
if (sym.range.Contains(target_line, target_column))
symbols.push_back(sym);
2017-06-15 05:32:23 +00:00
}
// Order shorter ranges first, since they are more detailed/precise. This is
// important for macros which generate code so that we can resolving the
// macro argument takes priority over the entire macro body.
//
2017-12-23 16:01:43 +00:00
// Order SymbolKind::Var before SymbolKind::Type. Macro calls are treated as
// Var currently. If a macro expands to tokens led by a SymbolKind::Type, the
// macro and the Type have the same range. We want to find the macro
// definition instead of the Type definition.
//
// Then order functions before other types, which makes goto definition work
// better on constructors.
2018-01-30 00:27:43 +00:00
std::sort(symbols.begin(), symbols.end(),
[](const SymbolRef& a, const SymbolRef& b) {
2018-02-09 17:42:10 +00:00
int a_size = ComputeRangeSize(a.range);
int b_size = ComputeRangeSize(b.range);
2018-01-30 00:27:43 +00:00
if (a_size != b_size)
return a_size < b_size;
// operator> orders Var/Func before Type.
2018-02-09 17:42:10 +00:00
int t = static_cast<int>(a.kind) - static_cast<int>(b.kind);
2018-01-30 00:27:43 +00:00
if (t)
return t > 0;
return a.id < b.id;
2018-01-30 00:27:43 +00:00
});
2017-06-15 05:32:23 +00:00
return symbols;
}
void EmitDiagnostics(WorkingFiles* working_files,
std::string path,
std::vector<lsDiagnostic> diagnostics) {
// Emit diagnostics.
Out_TextDocumentPublishDiagnostics out;
out.params.uri = lsDocumentUri::FromPath(path);
out.params.diagnostics = diagnostics;
QueueManager::WriteStdout(IpcId::TextDocumentPublishDiagnostics, out);
// Cache diagnostics so we can show fixits.
working_files->DoActionOnFile(path, [&](WorkingFile* working_file) {
if (working_file)
working_file->diagnostics_ = diagnostics;
});
}