2017-02-23 09:23:23 +00:00
|
|
|
#pragma once
|
|
|
|
|
2017-02-26 08:11:47 +00:00
|
|
|
#include "indexer.h"
|
2017-03-07 09:32:29 +00:00
|
|
|
#include "serializer.h"
|
2017-02-26 08:11:47 +00:00
|
|
|
|
2017-04-19 05:45:37 +00:00
|
|
|
#include <sparsepp/spp.h>
|
2017-04-08 01:35:12 +00:00
|
|
|
|
2018-02-18 05:52:14 +00:00
|
|
|
#include <forward_list>
|
2017-04-17 20:40:50 +00:00
|
|
|
#include <functional>
|
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryFile;
|
|
|
|
struct QueryType;
|
|
|
|
struct QueryFunc;
|
|
|
|
struct QueryVar;
|
|
|
|
struct QueryDatabase;
|
2017-04-07 06:57:26 +00:00
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
using QueryFileId = Id<QueryFile>;
|
|
|
|
using QueryTypeId = Id<QueryType>;
|
|
|
|
using QueryFuncId = Id<QueryFunc>;
|
|
|
|
using QueryVarId = Id<QueryVar>;
|
2017-04-07 06:57:26 +00:00
|
|
|
|
2018-02-04 00:20:14 +00:00
|
|
|
struct IdMap;
|
|
|
|
|
2017-02-26 08:11:47 +00:00
|
|
|
// There are two sources of reindex updates: the (single) definition of a
|
|
|
|
// symbol has changed, or one of many users of the symbol has changed.
|
|
|
|
//
|
|
|
|
// For simplicitly, if the single definition has changed, we update all of the
|
|
|
|
// associated single-owner definition data. See |Update*DefId|.
|
|
|
|
//
|
|
|
|
// If one of the many symbol users submits an update, we store the update such
|
|
|
|
// that it can be merged with other updates before actually being applied to
|
|
|
|
// the main database. See |MergeableUpdate|.
|
|
|
|
|
2017-09-22 01:14:57 +00:00
|
|
|
template <typename TId, typename TValue>
|
2017-02-26 08:11:47 +00:00
|
|
|
struct MergeableUpdate {
|
|
|
|
// The type/func/var which is getting new usages.
|
2017-04-15 05:24:49 +00:00
|
|
|
TId id;
|
2017-02-26 08:11:47 +00:00
|
|
|
// Entries to add and remove.
|
|
|
|
std::vector<TValue> to_add;
|
|
|
|
std::vector<TValue> to_remove;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2018-02-07 06:46:05 +00:00
|
|
|
MergeableUpdate(TId id, std::vector<TValue>&& to_add)
|
|
|
|
: id(id), to_add(std::move(to_add)) {}
|
2017-09-22 01:14:57 +00:00
|
|
|
MergeableUpdate(TId id,
|
2018-02-07 06:46:05 +00:00
|
|
|
std::vector<TValue>&& to_add,
|
|
|
|
std::vector<TValue>&& to_remove)
|
|
|
|
: id(id), to_add(std::move(to_add)), to_remove(std::move(to_remove)) {}
|
2017-02-27 07:23:43 +00:00
|
|
|
};
|
2017-09-22 01:14:57 +00:00
|
|
|
template <typename TVisitor, typename TId, typename TValue>
|
2017-05-19 07:02:01 +00:00
|
|
|
void Reflect(TVisitor& visitor, MergeableUpdate<TId, TValue>& value) {
|
2018-01-08 04:10:16 +00:00
|
|
|
REFLECT_MEMBER_START();
|
2017-05-19 07:02:01 +00:00
|
|
|
REFLECT_MEMBER(id);
|
|
|
|
REFLECT_MEMBER(to_add);
|
|
|
|
REFLECT_MEMBER(to_remove);
|
|
|
|
REFLECT_MEMBER_END();
|
|
|
|
}
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2017-12-12 07:56:50 +00:00
|
|
|
template <typename T>
|
|
|
|
struct WithUsr {
|
2018-01-13 08:10:39 +00:00
|
|
|
Usr usr;
|
2017-12-12 07:56:50 +00:00
|
|
|
T value;
|
|
|
|
|
2018-01-13 08:10:39 +00:00
|
|
|
WithUsr(Usr usr, const T& value) : usr(usr), value(value) {}
|
2018-02-06 07:22:44 +00:00
|
|
|
WithUsr(Usr usr, T&& value) : usr(usr), value(std::move(value)) {}
|
2017-12-12 07:56:50 +00:00
|
|
|
};
|
|
|
|
template <typename TVisitor, typename T>
|
|
|
|
void Reflect(TVisitor& visitor, WithUsr<T>& value) {
|
2018-01-08 04:10:16 +00:00
|
|
|
REFLECT_MEMBER_START();
|
2017-12-12 07:56:50 +00:00
|
|
|
REFLECT_MEMBER(usr);
|
|
|
|
REFLECT_MEMBER(value);
|
|
|
|
REFLECT_MEMBER_END();
|
|
|
|
}
|
|
|
|
|
2018-01-30 05:34:28 +00:00
|
|
|
template <typename T>
|
|
|
|
struct WithFileContent {
|
|
|
|
T value;
|
|
|
|
std::string file_content;
|
|
|
|
|
|
|
|
WithFileContent(const T& value, const std::string& file_content) : value(value), file_content(file_content) {}
|
|
|
|
};
|
|
|
|
template <typename TVisitor, typename T>
|
|
|
|
void Reflect(TVisitor& visitor, WithFileContent<T>& value) {
|
|
|
|
REFLECT_MEMBER_START();
|
|
|
|
REFLECT_MEMBER(value);
|
|
|
|
REFLECT_MEMBER(file_content);
|
|
|
|
REFLECT_MEMBER_END();
|
|
|
|
}
|
|
|
|
|
2018-02-10 08:06:45 +00:00
|
|
|
struct QueryFamily {
|
|
|
|
using FileId = Id<QueryFile>;
|
|
|
|
using FuncId = Id<QueryFunc>;
|
|
|
|
using TypeId = Id<QueryType>;
|
|
|
|
using VarId = Id<QueryVar>;
|
|
|
|
using Range = Reference;
|
|
|
|
};
|
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryFile {
|
2017-04-07 05:42:57 +00:00
|
|
|
struct Def {
|
2017-04-15 05:41:35 +00:00
|
|
|
std::string path;
|
2018-02-15 06:41:07 +00:00
|
|
|
std::vector<std::string> args;
|
2017-11-30 03:47:29 +00:00
|
|
|
// Language identifier
|
|
|
|
std::string language;
|
2017-05-21 03:46:15 +00:00
|
|
|
// Includes in the file.
|
|
|
|
std::vector<IndexInclude> includes;
|
2017-04-07 05:42:57 +00:00
|
|
|
// Outline of the file (ie, for code lens).
|
2017-04-08 07:11:57 +00:00
|
|
|
std::vector<SymbolRef> outline;
|
2017-04-07 05:42:57 +00:00
|
|
|
// Every symbol found in the file (ie, for goto definition)
|
2017-04-08 07:11:57 +00:00
|
|
|
std::vector<SymbolRef> all_symbols;
|
2017-10-28 22:09:14 +00:00
|
|
|
// Parts of the file which are disabled.
|
|
|
|
std::vector<Range> inactive_regions;
|
2018-01-27 17:29:28 +00:00
|
|
|
// Used by |$cquery/freshenIndex|.
|
|
|
|
std::vector<std::string> dependencies;
|
2017-04-07 05:42:57 +00:00
|
|
|
};
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2018-01-30 05:34:28 +00:00
|
|
|
using DefUpdate = WithFileContent<Def>;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2018-01-30 05:34:28 +00:00
|
|
|
optional<Def> def;
|
2018-02-02 06:47:22 +00:00
|
|
|
Maybe<Id<void>> symbol_idx;
|
2017-04-07 05:42:57 +00:00
|
|
|
|
2017-12-12 07:56:50 +00:00
|
|
|
explicit QueryFile(const std::string& path) {
|
2018-01-30 05:34:28 +00:00
|
|
|
def = Def();
|
2017-10-17 05:44:58 +00:00
|
|
|
def->path = path;
|
|
|
|
}
|
2017-02-26 08:11:47 +00:00
|
|
|
};
|
2017-10-28 22:09:14 +00:00
|
|
|
MAKE_REFLECT_STRUCT(QueryFile::Def,
|
|
|
|
path,
|
2018-02-15 06:41:07 +00:00
|
|
|
args,
|
2017-11-30 03:47:29 +00:00
|
|
|
language,
|
2017-10-28 22:09:14 +00:00
|
|
|
outline,
|
|
|
|
all_symbols,
|
2018-01-27 17:29:28 +00:00
|
|
|
inactive_regions,
|
|
|
|
dependencies);
|
2017-03-14 08:33:39 +00:00
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryType {
|
2018-02-10 08:06:45 +00:00
|
|
|
using Def = TypeDefDefinitionData<QueryFamily>;
|
2017-12-12 07:56:50 +00:00
|
|
|
using DefUpdate = WithUsr<Def>;
|
2017-04-15 05:24:49 +00:00
|
|
|
using DerivedUpdate = MergeableUpdate<QueryTypeId, QueryTypeId>;
|
2017-04-21 07:03:33 +00:00
|
|
|
using InstancesUpdate = MergeableUpdate<QueryTypeId, QueryVarId>;
|
2018-02-10 20:53:18 +00:00
|
|
|
using UsesUpdate = MergeableUpdate<QueryTypeId, Use>;
|
2017-02-27 02:03:14 +00:00
|
|
|
|
2018-01-13 08:10:39 +00:00
|
|
|
Usr usr;
|
2018-02-03 21:12:03 +00:00
|
|
|
Maybe<Id<void>> symbol_idx;
|
2018-02-18 07:09:05 +00:00
|
|
|
std::forward_list<Def> def;
|
2018-02-05 18:12:28 +00:00
|
|
|
std::vector<QueryTypeId> derived;
|
|
|
|
std::vector<QueryVarId> instances;
|
2018-02-10 20:53:18 +00:00
|
|
|
std::vector<Use> uses;
|
2017-02-26 08:11:47 +00:00
|
|
|
|
2018-02-08 03:14:44 +00:00
|
|
|
explicit QueryType(const Usr& usr) : usr(usr) {}
|
2018-02-18 07:09:05 +00:00
|
|
|
const Def* AnyDef() const {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
|
|
|
Def* AnyDef() {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
2017-02-26 08:11:47 +00:00
|
|
|
};
|
2017-03-14 08:33:39 +00:00
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryFunc {
|
2018-02-10 08:06:45 +00:00
|
|
|
using Def = FuncDefDefinitionData<QueryFamily>;
|
2017-12-12 07:56:50 +00:00
|
|
|
using DefUpdate = WithUsr<Def>;
|
2018-02-10 20:53:18 +00:00
|
|
|
using DeclarationsUpdate = MergeableUpdate<QueryFuncId, Use>;
|
2017-04-15 05:24:49 +00:00
|
|
|
using DerivedUpdate = MergeableUpdate<QueryFuncId, QueryFuncId>;
|
2018-02-10 20:53:18 +00:00
|
|
|
using UsesUpdate = MergeableUpdate<QueryFuncId, Use>;
|
2017-02-27 02:03:14 +00:00
|
|
|
|
2018-01-13 08:10:39 +00:00
|
|
|
Usr usr;
|
2018-02-03 21:12:03 +00:00
|
|
|
Maybe<Id<void>> symbol_idx;
|
2018-02-18 07:09:05 +00:00
|
|
|
std::forward_list<Def> def;
|
2018-02-10 20:53:18 +00:00
|
|
|
std::vector<Use> declarations;
|
2018-02-05 18:12:28 +00:00
|
|
|
std::vector<QueryFuncId> derived;
|
2018-02-10 20:53:18 +00:00
|
|
|
std::vector<Use> uses;
|
2017-02-26 08:11:47 +00:00
|
|
|
|
2018-02-08 03:14:44 +00:00
|
|
|
explicit QueryFunc(const Usr& usr) : usr(usr) {}
|
2018-02-18 07:09:05 +00:00
|
|
|
const Def* AnyDef() const {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
|
|
|
Def* AnyDef() {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
2017-02-26 08:11:47 +00:00
|
|
|
};
|
2017-03-14 08:33:39 +00:00
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryVar {
|
2018-02-10 08:06:45 +00:00
|
|
|
using Def = VarDefDefinitionData<QueryFamily>;
|
2017-12-12 07:56:50 +00:00
|
|
|
using DefUpdate = WithUsr<Def>;
|
2018-02-10 20:53:18 +00:00
|
|
|
using DeclarationsUpdate = MergeableUpdate<QueryVarId, Use>;
|
|
|
|
using UsesUpdate = MergeableUpdate<QueryVarId, Use>;
|
2017-02-26 08:11:47 +00:00
|
|
|
|
2017-12-12 07:56:50 +00:00
|
|
|
Usr usr;
|
2018-02-03 21:12:03 +00:00
|
|
|
Maybe<Id<void>> symbol_idx;
|
2018-02-18 05:52:14 +00:00
|
|
|
std::forward_list<Def> def;
|
2018-02-10 20:53:18 +00:00
|
|
|
std::vector<Use> declarations;
|
|
|
|
std::vector<Use> uses;
|
2017-02-26 08:11:47 +00:00
|
|
|
|
2018-02-08 03:14:44 +00:00
|
|
|
explicit QueryVar(const Usr& usr) : usr(usr) {}
|
2018-02-18 05:52:14 +00:00
|
|
|
const Def* AnyDef() const {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
|
|
|
Def* AnyDef() {
|
|
|
|
if (def.empty()) return nullptr;
|
|
|
|
return &def.front();
|
|
|
|
}
|
2017-02-26 08:11:47 +00:00
|
|
|
};
|
2017-03-14 08:33:39 +00:00
|
|
|
|
2017-02-27 07:23:43 +00:00
|
|
|
struct IndexUpdate {
|
2017-04-08 06:45:28 +00:00
|
|
|
// Creates a new IndexUpdate based on the delta from previous to current. If
|
|
|
|
// no delta computation should be done just pass null for previous.
|
2017-09-22 01:14:57 +00:00
|
|
|
static IndexUpdate CreateDelta(const IdMap* previous_id_map,
|
|
|
|
const IdMap* current_id_map,
|
|
|
|
IndexFile* previous,
|
|
|
|
IndexFile* current);
|
2017-04-07 05:42:57 +00:00
|
|
|
|
|
|
|
// Merge |update| into this update; this can reduce overhead / index update
|
|
|
|
// work can be parallelized.
|
2018-02-06 07:22:44 +00:00
|
|
|
void Merge(IndexUpdate&& update);
|
2017-04-07 05:42:57 +00:00
|
|
|
|
2017-05-19 07:02:01 +00:00
|
|
|
// Dump the update to a string.
|
|
|
|
std::string ToString();
|
|
|
|
|
2017-02-27 07:23:43 +00:00
|
|
|
// File updates.
|
2018-01-26 06:42:58 +00:00
|
|
|
std::vector<std::string> files_removed;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryFile::DefUpdate> files_def_update;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
|
|
|
// Type updates.
|
|
|
|
std::vector<Usr> types_removed;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryType::DefUpdate> types_def_update;
|
|
|
|
std::vector<QueryType::DerivedUpdate> types_derived;
|
2017-04-21 07:03:33 +00:00
|
|
|
std::vector<QueryType::InstancesUpdate> types_instances;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryType::UsesUpdate> types_uses;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
|
|
|
// Function updates.
|
|
|
|
std::vector<Usr> funcs_removed;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryFunc::DefUpdate> funcs_def_update;
|
|
|
|
std::vector<QueryFunc::DeclarationsUpdate> funcs_declarations;
|
|
|
|
std::vector<QueryFunc::DerivedUpdate> funcs_derived;
|
2018-02-10 20:53:18 +00:00
|
|
|
std::vector<QueryFunc::UsesUpdate> funcs_uses;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
|
|
|
// Variable updates.
|
|
|
|
std::vector<Usr> vars_removed;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryVar::DefUpdate> vars_def_update;
|
2018-01-27 05:50:17 +00:00
|
|
|
std::vector<QueryVar::DeclarationsUpdate> vars_declarations;
|
2017-04-15 05:41:35 +00:00
|
|
|
std::vector<QueryVar::UsesUpdate> vars_uses;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2017-04-07 05:42:57 +00:00
|
|
|
private:
|
2017-02-27 07:23:43 +00:00
|
|
|
// Creates an index update assuming that |previous| is already
|
|
|
|
// in the index, so only the delta between |previous| and |current|
|
|
|
|
// will be applied.
|
2017-09-22 01:14:57 +00:00
|
|
|
IndexUpdate(const IdMap& previous_id_map,
|
|
|
|
const IdMap& current_id_map,
|
|
|
|
IndexFile& previous,
|
|
|
|
IndexFile& current);
|
2017-02-27 07:23:43 +00:00
|
|
|
};
|
2017-09-22 01:14:57 +00:00
|
|
|
// NOTICE: We're not reflecting on files_removed or files_def_update, it is too
|
|
|
|
// much output when logging
|
2017-05-19 07:02:01 +00:00
|
|
|
MAKE_REFLECT_STRUCT(IndexUpdate,
|
2017-09-22 01:14:57 +00:00
|
|
|
types_removed,
|
|
|
|
types_def_update,
|
|
|
|
types_derived,
|
|
|
|
types_instances,
|
|
|
|
types_uses,
|
|
|
|
funcs_removed,
|
|
|
|
funcs_def_update,
|
|
|
|
funcs_declarations,
|
|
|
|
funcs_derived,
|
2018-02-10 20:53:18 +00:00
|
|
|
funcs_uses,
|
2017-09-22 01:14:57 +00:00
|
|
|
vars_removed,
|
|
|
|
vars_def_update,
|
2018-01-27 05:50:17 +00:00
|
|
|
vars_declarations,
|
2017-09-22 01:14:57 +00:00
|
|
|
vars_uses);
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2018-01-30 00:55:51 +00:00
|
|
|
struct NormalizedPath {
|
|
|
|
explicit NormalizedPath(const std::string& path);
|
|
|
|
bool operator==(const NormalizedPath& rhs) const;
|
|
|
|
bool operator!=(const NormalizedPath& rhs) const;
|
|
|
|
|
|
|
|
std::string path;
|
|
|
|
};
|
|
|
|
MAKE_HASHABLE(NormalizedPath, t.path);
|
|
|
|
|
2017-02-27 07:23:43 +00:00
|
|
|
// The query database is heavily optimized for fast queries. It is stored
|
|
|
|
// in-memory.
|
2017-04-15 05:41:35 +00:00
|
|
|
struct QueryDatabase {
|
2018-02-02 01:59:01 +00:00
|
|
|
// All File/Func/Type/Var symbols.
|
2017-02-27 07:23:43 +00:00
|
|
|
std::vector<SymbolIdx> symbols;
|
|
|
|
|
2017-04-22 07:32:29 +00:00
|
|
|
// Raw data storage. Accessible via SymbolIdx instances.
|
2017-10-17 05:44:58 +00:00
|
|
|
std::vector<QueryFile> files;
|
|
|
|
std::vector<QueryType> types;
|
|
|
|
std::vector<QueryFunc> funcs;
|
|
|
|
std::vector<QueryVar> vars;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
|
|
|
// Lookup symbol based on a usr.
|
2018-01-30 00:55:51 +00:00
|
|
|
spp::sparse_hash_map<NormalizedPath, QueryFileId> usr_to_file;
|
2017-05-09 04:20:28 +00:00
|
|
|
spp::sparse_hash_map<Usr, QueryTypeId> usr_to_type;
|
|
|
|
spp::sparse_hash_map<Usr, QueryFuncId> usr_to_func;
|
|
|
|
spp::sparse_hash_map<Usr, QueryVarId> usr_to_var;
|
2017-02-27 07:23:43 +00:00
|
|
|
|
2017-04-19 06:56:37 +00:00
|
|
|
// Marks the given Usrs as invalid.
|
2017-05-09 04:20:28 +00:00
|
|
|
void RemoveUsrs(SymbolKind usr_kind, const std::vector<Usr>& to_remove);
|
2017-02-27 07:23:43 +00:00
|
|
|
// Insert the contents of |update| into |db|.
|
|
|
|
void ApplyIndexUpdate(IndexUpdate* update);
|
2017-04-15 05:41:35 +00:00
|
|
|
void ImportOrUpdate(const std::vector<QueryFile::DefUpdate>& updates);
|
2018-02-07 05:26:38 +00:00
|
|
|
void ImportOrUpdate(std::vector<QueryType::DefUpdate>&& updates);
|
|
|
|
void ImportOrUpdate(std::vector<QueryFunc::DefUpdate>&& updates);
|
|
|
|
void ImportOrUpdate(std::vector<QueryVar::DefUpdate>&& updates);
|
2018-02-11 18:25:37 +00:00
|
|
|
void UpdateSymbols(Maybe<Id<void>>* symbol_idx, SymbolKind kind, Id<void> idx);
|
2018-02-03 18:33:22 +00:00
|
|
|
std::string_view GetSymbolDetailedName(RawId symbol_idx) const;
|
|
|
|
std::string_view GetSymbolShortName(RawId symbol_idx) const;
|
2018-01-20 16:57:41 +00:00
|
|
|
|
|
|
|
// Query the indexing structure to look up symbol id for given Usr.
|
2018-02-02 07:10:37 +00:00
|
|
|
Maybe<QueryFileId> GetQueryFileIdFromPath(const std::string& path);
|
|
|
|
Maybe<QueryTypeId> GetQueryTypeIdFromUsr(Usr usr);
|
|
|
|
Maybe<QueryFuncId> GetQueryFuncIdFromUsr(Usr usr);
|
|
|
|
Maybe<QueryVarId> GetQueryVarIdFromUsr(Usr usr);
|
2018-02-10 01:30:22 +00:00
|
|
|
|
2018-02-11 21:42:48 +00:00
|
|
|
Maybe<QueryFileId> GetFileId(SymbolIdx ref) {
|
2018-02-10 01:30:22 +00:00
|
|
|
switch (ref.kind) {
|
|
|
|
case SymbolKind::Invalid:
|
|
|
|
break;
|
|
|
|
case SymbolKind::File:
|
|
|
|
return QueryFileId(ref.id);
|
|
|
|
case SymbolKind::Func: {
|
2018-02-18 05:52:14 +00:00
|
|
|
const QueryFunc::Def* def = funcs[ref.id.id].AnyDef();
|
|
|
|
if (def)
|
|
|
|
return def->file;
|
2018-02-10 01:30:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SymbolKind::Type: {
|
2018-02-18 05:52:14 +00:00
|
|
|
const QueryType::Def* def = types[ref.id.id].AnyDef();
|
|
|
|
if (def)
|
|
|
|
return def->file;
|
2018-02-10 01:30:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SymbolKind::Var: {
|
2018-02-18 05:52:14 +00:00
|
|
|
const QueryVar::Def* def = vars[ref.id.id].AnyDef();
|
|
|
|
if (def)
|
|
|
|
return def->file;
|
2018-02-10 01:30:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QueryFileId();
|
|
|
|
}
|
2018-02-11 18:25:37 +00:00
|
|
|
QueryFile& GetFile(SymbolIdx ref) {
|
2018-02-10 01:30:22 +00:00
|
|
|
return files[ref.id.id];
|
|
|
|
}
|
2018-02-11 18:25:37 +00:00
|
|
|
QueryFunc& GetFunc(SymbolIdx ref) {
|
2018-02-10 01:30:22 +00:00
|
|
|
return funcs[ref.id.id];
|
|
|
|
}
|
2018-02-11 18:25:37 +00:00
|
|
|
QueryType& GetType(SymbolIdx ref) {
|
2018-02-10 01:30:22 +00:00
|
|
|
return types[ref.id.id];
|
|
|
|
}
|
2018-02-11 18:25:37 +00:00
|
|
|
QueryVar& GetVar(SymbolIdx ref) {
|
2018-02-10 01:30:22 +00:00
|
|
|
return vars[ref.id.id];
|
|
|
|
}
|
2017-04-07 08:01:58 +00:00
|
|
|
};
|
|
|
|
|
2018-02-04 02:16:30 +00:00
|
|
|
template <typename I>
|
|
|
|
struct IndexToQuery;
|
|
|
|
|
|
|
|
// clang-format off
|
2018-02-08 03:14:44 +00:00
|
|
|
template <> struct IndexToQuery<IndexFileId> { using type = QueryFileId; };
|
2018-02-04 02:16:30 +00:00
|
|
|
template <> struct IndexToQuery<IndexFuncId> { using type = QueryFuncId; };
|
|
|
|
template <> struct IndexToQuery<IndexTypeId> { using type = QueryTypeId; };
|
|
|
|
template <> struct IndexToQuery<IndexVarId> { using type = QueryVarId; };
|
2018-02-10 20:53:18 +00:00
|
|
|
template <> struct IndexToQuery<Use> { using type = Use; };
|
|
|
|
template <> struct IndexToQuery<SymbolRef> { using type = SymbolRef; };
|
|
|
|
template <> struct IndexToQuery<IndexFunc::Declaration> { using type = Use; };
|
2018-02-04 02:16:30 +00:00
|
|
|
template <typename I> struct IndexToQuery<optional<I>> {
|
|
|
|
using type = optional<typename IndexToQuery<I>::type>;
|
|
|
|
};
|
|
|
|
template <typename I> struct IndexToQuery<std::vector<I>> {
|
|
|
|
using type = std::vector<typename IndexToQuery<I>::type>;
|
|
|
|
};
|
|
|
|
// clang-format on
|
|
|
|
|
2017-04-07 08:01:58 +00:00
|
|
|
struct IdMap {
|
|
|
|
const IdCache& local_ids;
|
2017-04-08 07:21:00 +00:00
|
|
|
QueryFileId primary_file;
|
2017-04-07 08:01:58 +00:00
|
|
|
|
2017-04-15 05:41:35 +00:00
|
|
|
IdMap(QueryDatabase* query_db, const IdCache& local_ids);
|
2017-04-07 08:01:58 +00:00
|
|
|
|
2018-02-04 00:20:14 +00:00
|
|
|
// FIXME Too verbose
|
|
|
|
// clang-format off
|
2017-04-08 07:11:57 +00:00
|
|
|
QueryTypeId ToQuery(IndexTypeId id) const;
|
|
|
|
QueryFuncId ToQuery(IndexFuncId id) const;
|
|
|
|
QueryVarId ToQuery(IndexVarId id) const;
|
2018-02-10 20:53:18 +00:00
|
|
|
SymbolRef ToQuery(SymbolRef ref) const;
|
|
|
|
Use ToQuery(Reference ref) const;
|
|
|
|
Use ToQuery(Use ref) const;
|
|
|
|
Use ToQuery(IndexFunc::Declaration decl) const;
|
2018-02-04 02:16:30 +00:00
|
|
|
template <typename I>
|
|
|
|
Maybe<typename IndexToQuery<I>::type> ToQuery(Maybe<I> id) const {
|
|
|
|
if (!id)
|
|
|
|
return nullopt;
|
|
|
|
return ToQuery(*id);
|
|
|
|
}
|
|
|
|
template <typename I>
|
|
|
|
std::vector<typename IndexToQuery<I>::type> ToQuery(const std::vector<I>& a) const {
|
|
|
|
std::vector<typename IndexToQuery<I>::type> ret;
|
|
|
|
ret.reserve(a.size());
|
|
|
|
for (auto& x : a)
|
|
|
|
ret.push_back(ToQuery(x));
|
|
|
|
return ret;
|
|
|
|
}
|
2018-02-04 00:20:14 +00:00
|
|
|
// clang-format on
|
2017-04-08 07:52:57 +00:00
|
|
|
|
2017-09-22 01:14:57 +00:00
|
|
|
|
|
|
|
private:
|
2017-04-19 05:45:37 +00:00
|
|
|
spp::sparse_hash_map<IndexTypeId, QueryTypeId> cached_type_ids_;
|
|
|
|
spp::sparse_hash_map<IndexFuncId, QueryFuncId> cached_func_ids_;
|
|
|
|
spp::sparse_hash_map<IndexVarId, QueryVarId> cached_var_ids_;
|
2017-04-07 08:01:58 +00:00
|
|
|
};
|