2019-03-01 17:52:53 +08:00
|
|
|
//===-- TestIndex.cpp -------------------------------------------*- C++ -*-===//
|
2018-08-20 22:39:32 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-08-20 22:39:32 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "TestIndex.h"
|
2019-01-28 22:01:55 +08:00
|
|
|
#include "clang/Index/IndexSymbol.h"
|
|
|
|
#include "llvm/Support/Regex.h"
|
2018-08-20 22:39:32 +08:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace clangd {
|
|
|
|
|
2019-01-07 23:45:19 +08:00
|
|
|
Symbol symbol(llvm::StringRef QName) {
|
2018-08-20 22:39:32 +08:00
|
|
|
Symbol Sym;
|
|
|
|
Sym.ID = SymbolID(QName.str());
|
|
|
|
size_t Pos = QName.rfind("::");
|
2019-01-07 23:45:19 +08:00
|
|
|
if (Pos == llvm::StringRef::npos) {
|
2018-08-20 22:39:32 +08:00
|
|
|
Sym.Name = QName;
|
|
|
|
Sym.Scope = "";
|
|
|
|
} else {
|
|
|
|
Sym.Name = QName.substr(Pos + 2);
|
|
|
|
Sym.Scope = QName.substr(0, Pos + 2);
|
|
|
|
}
|
|
|
|
return Sym;
|
|
|
|
}
|
|
|
|
|
2019-01-28 22:01:55 +08:00
|
|
|
static std::string replace(llvm::StringRef Haystack, llvm::StringRef Needle,
|
|
|
|
llvm::StringRef Repl) {
|
2020-12-10 20:36:35 +08:00
|
|
|
llvm::SmallVector<llvm::StringRef> Parts;
|
2019-04-25 01:00:38 +08:00
|
|
|
Haystack.split(Parts, Needle);
|
|
|
|
return llvm::join(Parts, Repl);
|
2019-01-28 22:01:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Helpers to produce fake index symbols for memIndex() or completions().
|
|
|
|
// USRFormat is a regex replacement string for the unqualified part of the USR.
|
|
|
|
Symbol sym(llvm::StringRef QName, index::SymbolKind Kind,
|
|
|
|
llvm::StringRef USRFormat) {
|
|
|
|
Symbol Sym;
|
|
|
|
std::string USR = "c:"; // We synthesize a few simple cases of USRs by hand!
|
|
|
|
size_t Pos = QName.rfind("::");
|
|
|
|
if (Pos == llvm::StringRef::npos) {
|
|
|
|
Sym.Name = QName;
|
|
|
|
Sym.Scope = "";
|
|
|
|
} else {
|
|
|
|
Sym.Name = QName.substr(Pos + 2);
|
|
|
|
Sym.Scope = QName.substr(0, Pos + 2);
|
|
|
|
USR += "@N@" + replace(QName.substr(0, Pos), "::", "@N@"); // ns:: -> @N@ns
|
|
|
|
}
|
|
|
|
USR += llvm::Regex("^.*$").sub(USRFormat, Sym.Name); // e.g. func -> @F@func#
|
|
|
|
Sym.ID = SymbolID(USR);
|
|
|
|
Sym.SymInfo.Kind = Kind;
|
|
|
|
Sym.Flags |= Symbol::IndexedForCodeCompletion;
|
|
|
|
Sym.Origin = SymbolOrigin::Static;
|
|
|
|
return Sym;
|
|
|
|
}
|
|
|
|
|
|
|
|
Symbol func(llvm::StringRef Name) { // Assumes the function has no args.
|
|
|
|
return sym(Name, index::SymbolKind::Function, "@F@\\0#"); // no args
|
|
|
|
}
|
|
|
|
|
|
|
|
Symbol cls(llvm::StringRef Name) {
|
|
|
|
return sym(Name, index::SymbolKind::Class, "@S@\\0");
|
|
|
|
}
|
|
|
|
|
2021-12-10 09:14:15 +08:00
|
|
|
Symbol enm(llvm::StringRef Name) {
|
|
|
|
return sym(Name, index::SymbolKind::Enum, "@E@\\0");
|
|
|
|
}
|
|
|
|
|
2019-01-28 22:01:55 +08:00
|
|
|
Symbol var(llvm::StringRef Name) {
|
|
|
|
return sym(Name, index::SymbolKind::Variable, "@\\0");
|
|
|
|
}
|
|
|
|
|
|
|
|
Symbol ns(llvm::StringRef Name) {
|
|
|
|
return sym(Name, index::SymbolKind::Namespace, "@N@\\0");
|
|
|
|
}
|
|
|
|
|
2022-04-26 21:38:28 +08:00
|
|
|
Symbol conceptSym(llvm::StringRef Name) {
|
|
|
|
return sym(Name, index::SymbolKind::Concept, "@CT@\\0");
|
|
|
|
}
|
|
|
|
|
2022-06-07 00:19:32 +08:00
|
|
|
Symbol objcSym(llvm::StringRef Name, index::SymbolKind Kind,
|
|
|
|
llvm::StringRef USRPrefix) {
|
|
|
|
Symbol Sym;
|
|
|
|
std::string USR = USRPrefix.str() + Name.str();
|
|
|
|
Sym.Name = Name;
|
|
|
|
Sym.Scope = "";
|
|
|
|
Sym.ID = SymbolID(USR);
|
|
|
|
Sym.SymInfo.Kind = Kind;
|
|
|
|
Sym.SymInfo.Lang = index::SymbolLanguage::ObjC;
|
|
|
|
Sym.Flags |= Symbol::IndexedForCodeCompletion;
|
|
|
|
Sym.Origin = SymbolOrigin::Static;
|
|
|
|
return Sym;
|
|
|
|
}
|
|
|
|
|
|
|
|
Symbol objcClass(llvm::StringRef Name) {
|
|
|
|
return objcSym(Name, index::SymbolKind::Class, "objc(cs)");
|
|
|
|
}
|
|
|
|
|
|
|
|
Symbol objcProtocol(llvm::StringRef Name) {
|
|
|
|
return objcSym(Name, index::SymbolKind::Protocol, "objc(pl)");
|
|
|
|
}
|
|
|
|
|
[clangd] Factor out the data-swapping functionality from MemIndex/DexIndex.
Summary:
This is now handled by a wrapper class SwapIndex, so MemIndex/DexIndex can be
immutable and focus on their job.
Old and busted:
I have a MemIndex, which holds a shared_ptr<vector<Symbol*>>, which keeps the
symbol slab alive. I update by calling build(shared_ptr<vector<Symbol*>>).
New hotness: I have a SwapIndex, which holds a unique_ptr<SymbolIndex>, which
holds a MemIndex, which holds a shared_ptr<void>, which keeps backing
data alive.
I update by building a new MemIndex and calling SwapIndex::reset().
Reviewers: kbobyrev, ioeric
Subscribers: ilya-biryukov, ioeric, MaskRay, jkorous, mgrang, arphaman, kadircet, cfe-commits
Differential Revision: https://reviews.llvm.org/D51422
llvm-svn: 341318
2018-09-03 22:37:43 +08:00
|
|
|
SymbolSlab generateSymbols(std::vector<std::string> QualifiedNames) {
|
2018-08-20 22:39:32 +08:00
|
|
|
SymbolSlab::Builder Slab;
|
2019-01-07 23:45:19 +08:00
|
|
|
for (llvm::StringRef QName : QualifiedNames)
|
2018-08-20 22:39:32 +08:00
|
|
|
Slab.insert(symbol(QName));
|
[clangd] Factor out the data-swapping functionality from MemIndex/DexIndex.
Summary:
This is now handled by a wrapper class SwapIndex, so MemIndex/DexIndex can be
immutable and focus on their job.
Old and busted:
I have a MemIndex, which holds a shared_ptr<vector<Symbol*>>, which keeps the
symbol slab alive. I update by calling build(shared_ptr<vector<Symbol*>>).
New hotness: I have a SwapIndex, which holds a unique_ptr<SymbolIndex>, which
holds a MemIndex, which holds a shared_ptr<void>, which keeps backing
data alive.
I update by building a new MemIndex and calling SwapIndex::reset().
Reviewers: kbobyrev, ioeric
Subscribers: ilya-biryukov, ioeric, MaskRay, jkorous, mgrang, arphaman, kadircet, cfe-commits
Differential Revision: https://reviews.llvm.org/D51422
llvm-svn: 341318
2018-09-03 22:37:43 +08:00
|
|
|
return std::move(Slab).build();
|
2018-08-20 22:39:32 +08:00
|
|
|
}
|
|
|
|
|
[clangd] Factor out the data-swapping functionality from MemIndex/DexIndex.
Summary:
This is now handled by a wrapper class SwapIndex, so MemIndex/DexIndex can be
immutable and focus on their job.
Old and busted:
I have a MemIndex, which holds a shared_ptr<vector<Symbol*>>, which keeps the
symbol slab alive. I update by calling build(shared_ptr<vector<Symbol*>>).
New hotness: I have a SwapIndex, which holds a unique_ptr<SymbolIndex>, which
holds a MemIndex, which holds a shared_ptr<void>, which keeps backing
data alive.
I update by building a new MemIndex and calling SwapIndex::reset().
Reviewers: kbobyrev, ioeric
Subscribers: ilya-biryukov, ioeric, MaskRay, jkorous, mgrang, arphaman, kadircet, cfe-commits
Differential Revision: https://reviews.llvm.org/D51422
llvm-svn: 341318
2018-09-03 22:37:43 +08:00
|
|
|
SymbolSlab generateNumSymbols(int Begin, int End) {
|
2018-08-20 22:39:32 +08:00
|
|
|
std::vector<std::string> Names;
|
2022-02-01 18:14:07 +08:00
|
|
|
for (int I = Begin; I <= End; I++)
|
|
|
|
Names.push_back(std::to_string(I));
|
[clangd] Factor out the data-swapping functionality from MemIndex/DexIndex.
Summary:
This is now handled by a wrapper class SwapIndex, so MemIndex/DexIndex can be
immutable and focus on their job.
Old and busted:
I have a MemIndex, which holds a shared_ptr<vector<Symbol*>>, which keeps the
symbol slab alive. I update by calling build(shared_ptr<vector<Symbol*>>).
New hotness: I have a SwapIndex, which holds a unique_ptr<SymbolIndex>, which
holds a MemIndex, which holds a shared_ptr<void>, which keeps backing
data alive.
I update by building a new MemIndex and calling SwapIndex::reset().
Reviewers: kbobyrev, ioeric
Subscribers: ilya-biryukov, ioeric, MaskRay, jkorous, mgrang, arphaman, kadircet, cfe-commits
Differential Revision: https://reviews.llvm.org/D51422
llvm-svn: 341318
2018-09-03 22:37:43 +08:00
|
|
|
return generateSymbols(Names);
|
2018-08-20 22:39:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string getQualifiedName(const Symbol &Sym) {
|
2019-04-12 18:09:37 +08:00
|
|
|
return (Sym.Scope + Sym.Name + Sym.TemplateSpecializationArgs).str();
|
2018-08-20 22:39:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> match(const SymbolIndex &I,
|
|
|
|
const FuzzyFindRequest &Req, bool *Incomplete) {
|
|
|
|
std::vector<std::string> Matches;
|
|
|
|
bool IsIncomplete = I.fuzzyFind(Req, [&](const Symbol &Sym) {
|
|
|
|
Matches.push_back(clang::clangd::getQualifiedName(Sym));
|
|
|
|
});
|
|
|
|
if (Incomplete)
|
|
|
|
*Incomplete = IsIncomplete;
|
|
|
|
return Matches;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns qualified names of symbols with any of IDs in the index.
|
2019-01-07 23:45:19 +08:00
|
|
|
std::vector<std::string> lookup(const SymbolIndex &I,
|
|
|
|
llvm::ArrayRef<SymbolID> IDs) {
|
2018-08-20 22:39:32 +08:00
|
|
|
LookupRequest Req;
|
|
|
|
Req.IDs.insert(IDs.begin(), IDs.end());
|
|
|
|
std::vector<std::string> Results;
|
|
|
|
I.lookup(Req, [&](const Symbol &Sym) {
|
|
|
|
Results.push_back(getQualifiedName(Sym));
|
|
|
|
});
|
|
|
|
return Results;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace clangd
|
|
|
|
} // namespace clang
|