| //===--- Index.cpp -----------------------------------------------*- C++-*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "Index.h" |
| #include "llvm/ADT/StringExtras.h" |
| #include "llvm/Support/SHA1.h" |
| #include "llvm/Support/raw_ostream.h" |
| |
| namespace clang { |
| namespace clangd { |
| using namespace llvm; |
| |
| raw_ostream &operator<<(raw_ostream &OS, const SymbolLocation &L) { |
| if (!L) |
| return OS << "(none)"; |
| return OS << L.FileURI << "[" << L.Start.Line << ":" << L.Start.Column << "-" |
| << L.End.Line << ":" << L.End.Column << ")"; |
| } |
| |
| SymbolID::SymbolID(StringRef USR) |
| : HashValue(SHA1::hash(arrayRefFromStringRef(USR))) {} |
| |
| raw_ostream &operator<<(raw_ostream &OS, const SymbolID &ID) { |
| OS << toHex(toStringRef(ID.HashValue)); |
| return OS; |
| } |
| |
| std::string SymbolID::str() const { |
| std::string ID; |
| llvm::raw_string_ostream OS(ID); |
| OS << *this; |
| return OS.str(); |
| } |
| |
| void operator>>(StringRef Str, SymbolID &ID) { |
| std::string HexString = fromHex(Str); |
| assert(HexString.size() == ID.HashValue.size()); |
| std::copy(HexString.begin(), HexString.end(), ID.HashValue.begin()); |
| } |
| |
| raw_ostream &operator<<(raw_ostream &OS, SymbolOrigin O) { |
| if (O == SymbolOrigin::Unknown) |
| return OS << "unknown"; |
| constexpr static char Sigils[] = "ADSM4567"; |
| for (unsigned I = 0; I < sizeof(Sigils); ++I) |
| if (static_cast<uint8_t>(O) & 1u << I) |
| OS << Sigils[I]; |
| return OS; |
| } |
| |
| raw_ostream &operator<<(raw_ostream &OS, const Symbol &S) { |
| return OS << S.Scope << S.Name; |
| } |
| |
| double 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 { |
| auto It = std::lower_bound(Symbols.begin(), Symbols.end(), ID, |
| [](const Symbol &S, const SymbolID &I) { |
| return S.ID < I; |
| }); |
| 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, DenseSet<StringRef> &Strings, |
| BumpPtrAllocator &Arena) { |
| // Intern replaces V with a reference to the same string owned by the arena. |
| auto Intern = [&](StringRef &V) { |
| auto R = Strings.insert(V); |
| if (R.second) { // New entry added to the table, copy the string. |
| *R.first = V.copy(Arena); |
| } |
| V = *R.first; |
| }; |
| |
| // We need to copy every StringRef field onto the arena. |
| Intern(S.Name); |
| Intern(S.Scope); |
| Intern(S.CanonicalDeclaration.FileURI); |
| Intern(S.Definition.FileURI); |
| |
| Intern(S.Signature); |
| Intern(S.CompletionSnippetSuffix); |
| |
| if (S.Detail) { |
| // Copy values of StringRefs into arena. |
| auto *Detail = Arena.Allocate<Symbol::Details>(); |
| *Detail = *S.Detail; |
| // Intern the actual strings. |
| Intern(Detail->Documentation); |
| Intern(Detail->ReturnType); |
| Intern(Detail->IncludeHeader); |
| // Replace the detail pointer with our copy. |
| S.Detail = Detail; |
| } |
| } |
| |
| void SymbolSlab::Builder::insert(const Symbol &S) { |
| auto R = SymbolIndex.try_emplace(S.ID, Symbols.size()); |
| if (R.second) { |
| Symbols.push_back(S); |
| own(Symbols.back(), Strings, Arena); |
| } else { |
| auto &Copy = Symbols[R.first->second] = S; |
| own(Copy, Strings, Arena); |
| } |
| } |
| |
| SymbolSlab SymbolSlab::Builder::build() && { |
| Symbols = {Symbols.begin(), Symbols.end()}; // Force shrink-to-fit. |
| // Sort symbols so the slab can binary search over them. |
| std::sort(Symbols.begin(), Symbols.end(), |
| [](const Symbol &L, const Symbol &R) { return L.ID < R.ID; }); |
| // We may have unused strings from overwritten symbols. Build a new arena. |
| BumpPtrAllocator NewArena; |
| DenseSet<StringRef> Strings; |
| for (auto &S : Symbols) |
| own(S, Strings, NewArena); |
| return SymbolSlab(std::move(NewArena), std::move(Symbols)); |
| } |
| |
| } // namespace clangd |
| } // namespace clang |