2017-12-06 03:32:33 +00:00
|
|
|
#include "message_handler.h"
|
|
|
|
#include "query_utils.h"
|
|
|
|
|
2017-12-06 05:03:38 +00:00
|
|
|
namespace {
|
2017-12-06 04:39:44 +00:00
|
|
|
struct Ipc_CqueryBase : public IpcMessage<Ipc_CqueryBase> {
|
|
|
|
const static IpcId kIpcId = IpcId::CqueryBase;
|
|
|
|
lsRequestId id;
|
|
|
|
lsTextDocumentPositionParams params;
|
|
|
|
};
|
|
|
|
MAKE_REFLECT_STRUCT(Ipc_CqueryBase, id, params);
|
|
|
|
REGISTER_IPC_MESSAGE(Ipc_CqueryBase);
|
|
|
|
|
2017-12-06 03:32:33 +00:00
|
|
|
struct CqueryBaseHandler : BaseMessageHandler<Ipc_CqueryBase> {
|
|
|
|
void Run(Ipc_CqueryBase* request) override {
|
|
|
|
QueryFile* file;
|
|
|
|
if (!FindFileOrFail(db, request->id,
|
|
|
|
request->params.textDocument.uri.GetPath(), &file)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
WorkingFile* working_file =
|
|
|
|
working_files->GetFileByFilename(file->def->path);
|
|
|
|
|
|
|
|
Out_LocationList out;
|
|
|
|
out.id = request->id;
|
|
|
|
for (const SymbolRef& ref :
|
|
|
|
FindSymbolsAtLocation(working_file, file, request->params.position)) {
|
|
|
|
if (ref.idx.kind == SymbolKind::Type) {
|
|
|
|
QueryType& type = db->types[ref.idx.idx];
|
|
|
|
if (!type.def)
|
|
|
|
continue;
|
|
|
|
std::vector<QueryLocation> locations =
|
|
|
|
ToQueryLocation(db, type.def->parents);
|
|
|
|
out.result = GetLsLocations(db, working_files, locations);
|
|
|
|
} else if (ref.idx.kind == SymbolKind::Func) {
|
|
|
|
QueryFunc& func = db->funcs[ref.idx.idx];
|
|
|
|
optional<QueryLocation> location =
|
|
|
|
GetBaseDefinitionOrDeclarationSpelling(db, func);
|
|
|
|
if (!location)
|
|
|
|
continue;
|
|
|
|
optional<lsLocation> ls_loc =
|
|
|
|
GetLsLocation(db, working_files, *location);
|
|
|
|
if (!ls_loc)
|
|
|
|
continue;
|
|
|
|
out.result.push_back(*ls_loc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
IpcManager::WriteStdout(IpcId::CqueryBase, out);
|
|
|
|
}
|
|
|
|
};
|
2017-12-06 05:03:38 +00:00
|
|
|
REGISTER_MESSAGE_HANDLER(CqueryBaseHandler);
|
|
|
|
} // namespace
|