2017-12-06 03:32:33 +00:00
|
|
|
#include "message_handler.h"
|
|
|
|
#include "query_utils.h"
|
2017-12-29 16:29:47 +00:00
|
|
|
#include "queue_manager.h"
|
2017-12-06 03:32:33 +00:00
|
|
|
|
|
|
|
namespace {
|
2017-12-12 05:21:03 +00:00
|
|
|
void PushBack(std::vector<lsLocation>* result, optional<lsLocation> location) {
|
2017-12-06 03:32:33 +00:00
|
|
|
if (location)
|
|
|
|
result->push_back(*location);
|
|
|
|
}
|
|
|
|
|
2017-12-06 04:39:44 +00:00
|
|
|
struct Ipc_TextDocumentDefinition
|
2018-01-19 08:47:52 +00:00
|
|
|
: public RequestMessage<Ipc_TextDocumentDefinition> {
|
2017-12-06 04:39:44 +00:00
|
|
|
const static IpcId kIpcId = IpcId::TextDocumentDefinition;
|
|
|
|
lsTextDocumentPositionParams params;
|
|
|
|
};
|
|
|
|
MAKE_REFLECT_STRUCT(Ipc_TextDocumentDefinition, id, params);
|
|
|
|
REGISTER_IPC_MESSAGE(Ipc_TextDocumentDefinition);
|
|
|
|
|
|
|
|
struct Out_TextDocumentDefinition
|
|
|
|
: public lsOutMessage<Out_TextDocumentDefinition> {
|
|
|
|
lsRequestId id;
|
2017-12-12 05:20:29 +00:00
|
|
|
std::vector<lsLocation> result;
|
2017-12-06 04:39:44 +00:00
|
|
|
};
|
|
|
|
MAKE_REFLECT_STRUCT(Out_TextDocumentDefinition, jsonrpc, id, result);
|
|
|
|
|
2018-02-09 05:11:35 +00:00
|
|
|
std::vector<Reference> GetGotoDefinitionTargets(QueryDatabase* db,
|
2018-02-09 17:42:10 +00:00
|
|
|
SymbolRef sym) {
|
|
|
|
switch (sym.kind) {
|
2017-12-20 06:20:44 +00:00
|
|
|
// Returns GetDeclarationsOfSymbolForGotoDefinition and
|
|
|
|
// variable type definition.
|
|
|
|
case SymbolKind::Var: {
|
2018-02-09 05:11:35 +00:00
|
|
|
std::vector<Reference> ret =
|
2018-02-09 17:42:10 +00:00
|
|
|
GetDeclarationsOfSymbolForGotoDefinition(db, sym);
|
|
|
|
QueryVar& var = sym.Var(db);
|
2017-12-20 06:20:44 +00:00
|
|
|
if (var.def && var.def->variable_type) {
|
2018-02-09 05:11:35 +00:00
|
|
|
std::vector<Reference> types = GetDeclarationsOfSymbolForGotoDefinition(
|
2018-02-09 17:42:10 +00:00
|
|
|
db, SymbolRef(Range(), Id<void>(var.def->variable_type->id),
|
|
|
|
SymbolKind::Type, SymbolRole::None));
|
2017-12-20 06:20:44 +00:00
|
|
|
ret.insert(ret.end(), types.begin(), types.end());
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
default:
|
2018-02-09 17:42:10 +00:00
|
|
|
return GetDeclarationsOfSymbolForGotoDefinition(db, sym);
|
2017-12-20 06:20:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-06 03:32:33 +00:00
|
|
|
struct TextDocumentDefinitionHandler
|
|
|
|
: BaseMessageHandler<Ipc_TextDocumentDefinition> {
|
|
|
|
void Run(Ipc_TextDocumentDefinition* request) override {
|
|
|
|
QueryFileId file_id;
|
|
|
|
QueryFile* file;
|
2017-12-31 03:18:33 +00:00
|
|
|
if (!FindFileOrFail(db, project, request->id,
|
2017-12-06 03:32:33 +00:00
|
|
|
request->params.textDocument.uri.GetPath(), &file,
|
|
|
|
&file_id)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
WorkingFile* working_file =
|
|
|
|
working_files->GetFileByFilename(file->def->path);
|
|
|
|
|
|
|
|
Out_TextDocumentDefinition out;
|
|
|
|
out.id = request->id;
|
|
|
|
|
2018-01-14 19:39:29 +00:00
|
|
|
int target_line = request->params.position.line;
|
|
|
|
int target_column = request->params.position.character;
|
2017-12-06 03:32:33 +00:00
|
|
|
|
2018-02-09 17:42:10 +00:00
|
|
|
for (const SymbolRef& sym :
|
2017-12-06 03:32:33 +00:00
|
|
|
FindSymbolsAtLocation(working_file, file, request->params.position)) {
|
|
|
|
// Found symbol. Return definition.
|
|
|
|
|
|
|
|
// Special cases which are handled:
|
|
|
|
// - symbol has declaration but no definition (ie, pure virtual)
|
|
|
|
// - start at spelling but end at extent for better mouse tooltip
|
|
|
|
// - goto declaration while in definition of recursive type
|
|
|
|
|
2018-02-10 01:30:22 +00:00
|
|
|
optional<Reference> def_loc = GetDefinitionSpellingOfSymbol(db, sym);
|
2017-12-06 03:32:33 +00:00
|
|
|
|
|
|
|
// We use spelling start and extent end because this causes vscode to
|
|
|
|
// highlight the entire definition when previewing / hoving with the
|
|
|
|
// mouse.
|
2018-02-10 01:30:22 +00:00
|
|
|
optional<Reference> def_extent = GetDefinitionExtentOfSymbol(db, sym);
|
2017-12-06 03:32:33 +00:00
|
|
|
if (def_loc && def_extent)
|
|
|
|
def_loc->range.end = def_extent->range.end;
|
|
|
|
|
|
|
|
// If the cursor is currently at or in the definition we should goto
|
|
|
|
// the declaration if possible. We also want to use declarations if
|
|
|
|
// we're pointing to, ie, a pure virtual function which has no
|
|
|
|
// definition.
|
2018-02-10 01:30:22 +00:00
|
|
|
if (!def_loc || (db->GetFileId(*def_loc) == file_id &&
|
2017-12-06 03:32:33 +00:00
|
|
|
def_loc->range.Contains(target_line, target_column))) {
|
|
|
|
// Goto declaration.
|
|
|
|
|
2018-02-09 17:42:10 +00:00
|
|
|
std::vector<Reference> targets = GetGotoDefinitionTargets(db, sym);
|
2018-02-09 05:11:35 +00:00
|
|
|
for (Reference target : targets) {
|
2017-12-20 06:20:44 +00:00
|
|
|
optional<lsLocation> ls_target =
|
|
|
|
GetLsLocation(db, working_files, target);
|
|
|
|
if (ls_target)
|
|
|
|
out.result.push_back(*ls_target);
|
2017-12-06 03:32:33 +00:00
|
|
|
}
|
|
|
|
// We found some declarations. Break so we don't add the definition
|
|
|
|
// location.
|
|
|
|
if (!out.result.empty())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def_loc) {
|
|
|
|
PushBack(&out.result, GetLsLocation(db, working_files, *def_loc));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!out.result.empty())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// No symbols - check for includes.
|
|
|
|
if (out.result.empty()) {
|
|
|
|
for (const IndexInclude& include : file->def->includes) {
|
|
|
|
if (include.line == target_line) {
|
|
|
|
lsLocation result;
|
|
|
|
result.uri = lsDocumentUri::FromPath(include.resolved_path);
|
|
|
|
out.result.push_back(result);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-24 00:25:18 +00:00
|
|
|
QueueManager::WriteStdout(IpcId::TextDocumentDefinition, out);
|
2017-12-06 03:32:33 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
REGISTER_MESSAGE_HANDLER(TextDocumentDefinitionHandler);
|
2017-12-20 06:20:44 +00:00
|
|
|
} // namespace
|