2019-02-28 21:23:03 +08:00
|
|
|
//===--- Symbol.cpp ----------------------------------------------*- C++-*-===//
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Symbol.h"
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace clangd {
|
|
|
|
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, Symbol::SymbolFlag F) {
|
|
|
|
if (F == Symbol::None)
|
|
|
|
return OS << "None";
|
|
|
|
std::string S;
|
|
|
|
if (F & Symbol::Deprecated)
|
|
|
|
S += "deprecated|";
|
|
|
|
if (F & Symbol::IndexedForCodeCompletion)
|
|
|
|
S += "completion|";
|
|
|
|
return OS << llvm::StringRef(S).rtrim('|');
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Symbol &S) {
|
|
|
|
return OS << S.Scope << S.Name;
|
|
|
|
}
|
|
|
|
|
|
|
|
float quality(const Symbol &S) {
|
|
|
|
// This avoids a sharp gradient for tail symbols, and also neatly avoids the
|
|
|
|
// question of whether 0 references means a bad symbol or missing data.
|
|
|
|
if (S.References < 3)
|
|
|
|
return 1;
|
|
|
|
return std::log(S.References);
|
|
|
|
}
|
|
|
|
|
|
|
|
SymbolSlab::const_iterator SymbolSlab::find(const SymbolID &ID) const {
|
2019-06-30 19:19:56 +08:00
|
|
|
auto It = llvm::partition_point(Symbols,
|
|
|
|
[&](const Symbol &S) { return S.ID < ID; });
|
2019-02-28 21:23:03 +08:00
|
|
|
if (It != Symbols.end() && It->ID == ID)
|
|
|
|
return It;
|
|
|
|
return Symbols.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the underlying data of the symbol into the owned arena.
|
|
|
|
static void own(Symbol &S, llvm::UniqueStringSaver &Strings) {
|
|
|
|
visitStrings(S, [&](llvm::StringRef &V) { V = Strings.save(V); });
|
|
|
|
}
|
|
|
|
|
|
|
|
void SymbolSlab::Builder::insert(const Symbol &S) {
|
2019-05-03 21:17:29 +08:00
|
|
|
own(Symbols[S.ID] = S, UniqueStrings);
|
2019-02-28 21:23:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SymbolSlab SymbolSlab::Builder::build() && {
|
2019-05-03 21:17:29 +08:00
|
|
|
// Sort symbols into vector so the slab can binary search over them.
|
|
|
|
std::vector<Symbol> SortedSymbols;
|
|
|
|
SortedSymbols.reserve(Symbols.size());
|
|
|
|
for (auto &Entry : Symbols)
|
|
|
|
SortedSymbols.push_back(std::move(Entry.second));
|
|
|
|
llvm::sort(SortedSymbols,
|
2019-02-28 21:23:03 +08:00
|
|
|
[](const Symbol &L, const Symbol &R) { return L.ID < R.ID; });
|
|
|
|
// We may have unused strings from overwritten symbols. Build a new arena.
|
|
|
|
llvm::BumpPtrAllocator NewArena;
|
|
|
|
llvm::UniqueStringSaver Strings(NewArena);
|
2019-05-03 21:17:29 +08:00
|
|
|
for (auto &S : SortedSymbols)
|
2019-02-28 21:23:03 +08:00
|
|
|
own(S, Strings);
|
2019-05-03 21:17:29 +08:00
|
|
|
return SymbolSlab(std::move(NewArena), std::move(SortedSymbols));
|
2019-02-28 21:23:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace clangd
|
|
|
|
} // namespace clang
|