22#include "clang/Tooling/CompilationDatabase.h"
23#include "llvm/ADT/StringRef.h"
24#include "llvm/Support/Allocator.h"
25#include "llvm/Support/StringSaver.h"
26#include "llvm/Support/YAMLTraits.h"
27#include "llvm/Support/raw_ostream.h"
32struct YIncludeHeaderWithReferences;
37LLVM_YAML_IS_SEQUENCE_VECTOR(YIncludeHeaderWithReferences)
41 std::pair<clang::clangd::SymbolID, std::vector<clang::clangd::Ref>>;
44 std::optional<clang::clangd::Symbol> Symbol;
45 std::optional<RefBundle> Refs;
46 std::optional<clang::clangd::Relation> Relation;
47 std::optional<clang::clangd::IncludeGraphNode> Source;
48 std::optional<clang::tooling::CompileCommand> Cmd;
58struct YIncludeHeaderWithReferences {
59 llvm::StringRef IncludeHeader;
63 YIncludeHeaderWithReferences() =
default;
65 YIncludeHeaderWithReferences(
66 llvm::StringRef IncludeHeader, uint32_t References,
68 : IncludeHeader(IncludeHeader), References(References),
69 SupportedDirectives(SupportedDirectives) {}
73struct CompileCommandYAML : clang::tooling::CompileCommand {};
81using clang::clangd::IncludeGraphNode;
82using clang::clangd::Ref;
84using clang::clangd::Relation;
86using clang::clangd::Symbol;
87using clang::clangd::SymbolID;
88using clang::clangd::SymbolLocation;
89using clang::index::SymbolInfo;
90using clang::index::SymbolKind;
91using clang::index::SymbolLanguage;
92using clang::tooling::CompileCommand;
105 I.setError(llvm::toString(ID.takeError()));
117 Flag =
static_cast<uint8_t
>(F);
127template <>
struct MappingTraits<YPosition> {
128 static void mapping(IO &IO, YPosition &Value) {
129 IO.mapRequired(
"Line", Value.Line);
130 IO.mapRequired(
"Column", Value.Column);
156 assert(IO.getContext() &&
157 "Expecting an UniqueStringSaver to allocate data");
158 return static_cast<llvm::UniqueStringSaver *
>(IO.getContext())
168 MappingNormalization<NormalizedFileURI, const char *> NFile(IO,
170 IO.mapRequired(
"FileURI", NFile->URI);
171 MappingNormalization<NormalizedPosition, SymbolLocation::Position> NStart(
173 IO.mapRequired(
"Start", NStart->P);
174 MappingNormalization<NormalizedPosition, SymbolLocation::Position> NEnd(
176 IO.mapRequired(
"End", NEnd->P);
183 IO.mapRequired(
"Kind", SymInfo.Kind);
184 IO.mapRequired(
"Lang", SymInfo.Lang);
188template <>
struct ScalarBitSetTraits<
clang::clangd::Symbol::IncludeDirective> {
195template <>
struct MappingTraits<YIncludeHeaderWithReferences> {
196 static void mapping(IO &IO, YIncludeHeaderWithReferences &Inc) {
197 IO.mapRequired(
"Header", Inc.IncludeHeader);
198 IO.mapRequired(
"References", Inc.References);
199 IO.mapOptional(
"Directives", Inc.SupportedDirectives,
208 IO &,
const llvm::SmallVector<IncludeHeader, 1> &IncludeHeaders) {
209 for (
auto &I : IncludeHeaders) {
210 Headers.emplace_back(I.IncludeHeader, I.References,
211 I.supportedDirectives());
216 llvm::SmallVector<IncludeHeader, 1> Result;
218 Result.emplace_back(H.IncludeHeader, H.References, H.SupportedDirectives);
221 llvm::SmallVector<YIncludeHeaderWithReferences, 1>
Headers;
224template <>
struct MappingTraits<
Symbol> {
226 MappingNormalization<NormalizedSymbolID, SymbolID> NSymbolID(IO, Sym.
ID);
227 MappingNormalization<NormalizedSymbolFlag, Symbol::SymbolFlag> NSymbolFlag(
229 MappingNormalization<
231 llvm::SmallVector<Symbol::IncludeHeaderWithReferences, 1>>
233 IO.mapRequired(
"ID", NSymbolID->HexString);
234 IO.mapRequired(
"Name", Sym.
Name);
235 IO.mapRequired(
"Scope", Sym.
Scope);
236 IO.mapRequired(
"SymInfo", Sym.
SymInfo);
240 IO.mapOptional(
"References", Sym.
References, 0u);
241 IO.mapOptional(
"Flags", NSymbolFlag->Flag);
242 IO.mapOptional(
"Signature", Sym.
Signature);
243 IO.mapOptional(
"TemplateSpecializationArgs",
248 IO.mapOptional(
"Type", Sym.
Type);
249 IO.mapOptional(
"IncludeHeaders", NIncludeHeaders->Headers);
253template <>
struct ScalarEnumerationTraits<SymbolLanguage> {
255 IO.enumCase(Value,
"C", SymbolLanguage::C);
256 IO.enumCase(Value,
"Cpp", SymbolLanguage::CXX);
257 IO.enumCase(Value,
"ObjC", SymbolLanguage::ObjC);
258 IO.enumCase(Value,
"Swift", SymbolLanguage::Swift);
264#define DEFINE_ENUM(name) IO.enumCase(Value, #name, SymbolKind::name)
299template <>
struct MappingTraits<RefBundle> {
300 static void mapping(IO &IO, RefBundle &Refs) {
301 MappingNormalization<NormalizedSymbolID, SymbolID> NSymbolID(IO,
303 IO.mapRequired(
"ID", NSymbolID->HexString);
304 IO.mapRequired(
"References", Refs.second);
317template <>
struct MappingTraits<
Ref> {
319 MappingNormalization<NormalizedRefKind, RefKind> NKind(IO, R.
Kind);
320 IO.mapRequired(
"Kind", NKind->Kind);
321 IO.mapRequired(
"Location", R.
Location);
322 IO.mapOptional(
"Container", R.
Container);
329 Kind =
static_cast<uint8_t
>(R);
339 MappingNormalization<NormalizedSymbolID, SymbolID> NSymbolID(IO, ID);
340 IO.mapRequired(
"ID", NSymbolID->HexString);
346 MappingNormalization<NormalizedSymbolRole, RelationKind> NRole(
349 IO.mapRequired(
"Predicate", NRole->Kind);
357 Flag =
static_cast<uint8_t
>(O);
375 if (
HexString.size() == Digest.size() * 2 &&
376 llvm::all_of(
HexString, llvm::isHexDigit)) {
377 memcpy(Digest.data(), llvm::fromHex(
HexString).data(), Digest.size());
379 I.setError(std::string(
"Bad hex file digest: ") +
HexString);
389 IO.mapRequired(
"URI", Node.
URI);
390 MappingNormalization<NormalizedSourceFlag, IncludeGraphNode::SourceFlag>
391 NSourceFlag(IO, Node.
Flags);
392 IO.mapRequired(
"Flags", NSourceFlag->Flag);
393 MappingNormalization<NormalizedFileDigest, FileDigest> NDigest(IO,
395 IO.mapRequired(
"Digest", NDigest->HexString);
400template <>
struct MappingTraits<CompileCommandYAML> {
401 static void mapping(IO &IO, CompileCommandYAML &Cmd) {
402 IO.mapRequired(
"Directory", Cmd.Directory);
403 IO.mapRequired(
"CommandLine", Cmd.CommandLine);
407template <>
struct MappingTraits<VariantEntry> {
408 static void mapping(IO &IO, VariantEntry &Variant) {
409 if (IO.mapTag(
"!Symbol", Variant.Symbol.has_value())) {
410 if (!IO.outputting())
411 Variant.Symbol.emplace();
412 MappingTraits<Symbol>::mapping(IO, *Variant.Symbol);
413 }
else if (IO.mapTag(
"!Refs", Variant.Refs.has_value())) {
414 if (!IO.outputting())
415 Variant.Refs.emplace();
416 MappingTraits<RefBundle>::mapping(IO, *Variant.Refs);
417 }
else if (IO.mapTag(
"!Relations", Variant.Relation.has_value())) {
418 if (!IO.outputting())
419 Variant.Relation.emplace();
420 MappingTraits<Relation>::mapping(IO, *Variant.Relation);
421 }
else if (IO.mapTag(
"!Source", Variant.Source.has_value())) {
422 if (!IO.outputting())
423 Variant.Source.emplace();
424 MappingTraits<IncludeGraphNode>::mapping(IO, *Variant.Source);
425 }
else if (IO.mapTag(
"!Cmd", Variant.Cmd.has_value())) {
426 if (!IO.outputting())
427 Variant.Cmd.emplace();
428 MappingTraits<CompileCommandYAML>::mapping(
429 IO,
static_cast<CompileCommandYAML &
>(*Variant.Cmd));
441 llvm::yaml::Output Yout(OS);
442 for (
const auto &Sym : *O.
Symbols) {
448 for (
auto &Sym : *O.
Refs) {
460 for (
const auto &Source : *O.
Sources) {
462 Entry.Source = Source.getValue();
473llvm::Expected<IndexFileIn>
readYAML(llvm::StringRef Data,
478 llvm::BumpPtrAllocator
480 llvm::UniqueStringSaver Strings(Arena);
481 llvm::yaml::Input Yin(Data, &Strings);
483 std::optional<tooling::CompileCommand> Cmd;
484 while (Yin.setCurrentDocument()) {
485 llvm::yaml::EmptyContext Ctx;
486 VariantEntry Variant;
487 yamlize(Yin, Variant,
true, Ctx);
489 return llvm::errorCodeToError(Yin.error());
491 if (Variant.Symbol) {
492 Variant.Symbol->Origin = Origin;
493 Symbols.insert(*Variant.Symbol);
496 for (
const auto &
Ref : Variant.Refs->second)
497 Refs.insert(Variant.Refs->first,
Ref);
498 if (Variant.Relation)
499 Relations.insert(*Variant.Relation);
500 if (Variant.Source) {
501 auto &IGN = *Variant.Source;
502 auto Entry = Sources.try_emplace(IGN.URI).first;
503 Entry->getValue() = std::move(IGN);
506 for (
auto &Include :
Entry->getValue().DirectIncludes)
507 Include = Sources.try_emplace(Include).first->getKey();
516 Result.
Refs.emplace(std::move(Refs).build());
517 Result.
Relations.emplace(std::move(Relations).build());
519 Result.
Sources = std::move(Sources);
520 Result.
Cmd = std::move(Cmd);
521 return std::move(Result);
527 llvm::raw_string_ostream OS(Buf);
528 llvm::yaml::Output Yout(OS);
536 RefBundle Refs = {Data.first, Data.second};
539 llvm::raw_string_ostream OS(Buf);
540 llvm::yaml::Output Yout(OS);
549 llvm::raw_string_ostream OS(Buf);
550 llvm::yaml::Output Yout(OS);
560 llvm::raw_string_ostream OS(Buf);
561 llvm::yaml::Output Yout(OS);
clang::find_all_symbols::SymbolInfo SymbolInfo
clang::find_all_symbols::SymbolInfo::SymbolKind SymbolKind
#define DEFINE_ENUM(name)
RefSlab::Builder is a mutable container that can 'freeze' to RefSlab.
RelationSlab::Builder is a mutable container that can 'freeze' to RelationSlab.
SymbolSlab::Builder is a mutable container that can 'freeze' to SymbolSlab.
static llvm::Expected< SymbolID > fromStr(llvm::StringRef)
FIXME: Skip testing on windows temporarily due to the different escaping code mode.
llvm::Expected< IndexFileIn > readYAML(llvm::StringRef, SymbolOrigin Origin)
llvm::StringMap< IncludeGraphNode > IncludeGraph
std::string toYAML(const Symbol &)
RefKind
Describes the kind of a cross-reference.
void writeYAML(const IndexFileOut &, llvm::raw_ostream &)
std::array< uint8_t, 8 > FileDigest
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
Some operations such as code completion produce a set of candidates.
std::vector< llvm::StringRef > DirectIncludes
std::optional< RelationSlab > Relations
std::optional< SymbolSlab > Symbols
std::optional< RefSlab > Refs
std::optional< tooling::CompileCommand > Cmd
std::optional< IncludeGraph > Sources
const IncludeGraph * Sources
const RelationSlab * Relations
const tooling::CompileCommand * Cmd
const SymbolSlab * Symbols
Represents a symbol occurrence in the source file.
SymbolID Container
The ID of the symbol whose definition contains this reference.
SymbolLocation Location
The source location where the symbol is named.
Represents a relation between two symbols.
void setColumn(uint32_t Column)
void setLine(uint32_t Line)
The class presents a C++ symbol, e.g.
@ Include
#include "header.h"
@ Import
#import "header.h"
SymbolLocation Definition
The location of the symbol's definition, if one was found.
llvm::StringRef Type
Raw representation of the OpaqueType of the symbol, used for scoring purposes.
llvm::StringRef Documentation
Documentation including comment for the symbol declaration.
index::SymbolInfo SymInfo
The symbol information, like symbol kind.
llvm::SmallVector< IncludeHeaderWithReferences, 1 > IncludeHeaders
One Symbol can potentially be included via different headers.
llvm::StringRef Name
The unqualified name of the symbol, e.g. "bar" (for ns::bar).
llvm::StringRef Scope
The containing namespace. e.g. "" (global), "ns::" (top-level namespace).
llvm::StringRef Signature
A brief description of the symbol that can be appended in the completion candidate list.
unsigned References
The number of translation units that reference this symbol from their main file.
llvm::StringRef ReturnType
Type when this symbol is used in an expression.
llvm::StringRef TemplateSpecializationArgs
Argument list in human-readable format, will be displayed to help disambiguate between different spec...
SymbolLocation CanonicalDeclaration
The location of the preferred declaration of the symbol.
llvm::StringRef CompletionSnippetSuffix
What to insert when completing this symbol, after the symbol name.
SymbolID ID
The ID of the symbol.
static void mapping(IO &IO, CompileCommandYAML &Cmd)
static void mapping(IO &IO, IncludeGraphNode &Node)
static void mapping(IO &IO, RefBundle &Refs)
static void mapping(IO &IO, Ref &R)
static void mapping(IO &IO, Relation &Relation)
static void mapping(IO &IO, SymbolID &ID)
static void mapping(IO &IO, SymbolInfo &SymInfo)
static void mapping(IO &IO, SymbolLocation &Value)
static void mapping(IO &IO, Symbol &Sym)
static void mapping(IO &IO, VariantEntry &Variant)
static void mapping(IO &IO, YPosition &Value)
NormalizedFileDigest(IO &, const FileDigest &Digest)
NormalizedFileDigest(IO &)
FileDigest denormalize(IO &I)
NormalizedFileURI(IO &, const char *FileURI)
const char * denormalize(IO &IO)
NormalizedPosition(IO &, const Position &Pos)
Position denormalize(IO &)
clang::clangd::SymbolLocation::Position Position
NormalizedRefKind(IO &, RefKind O)
RefKind denormalize(IO &)
IncludeGraphNode::SourceFlag denormalize(IO &)
NormalizedSourceFlag(IO &)
NormalizedSourceFlag(IO &, IncludeGraphNode::SourceFlag O)
NormalizedSymbolFlag(IO &, Symbol::SymbolFlag F)
Symbol::SymbolFlag denormalize(IO &)
NormalizedSymbolFlag(IO &)
SymbolID denormalize(IO &I)
NormalizedSymbolID(IO &, const SymbolID &ID)
NormalizedSymbolRole(IO &IO, RelationKind R)
RelationKind denormalize(IO &IO)
NormalizedSymbolRole(IO &)
static void bitset(IO &IO, clang::clangd::Symbol::IncludeDirective &Value)
static void enumeration(IO &IO, SymbolKind &Value)
static void enumeration(IO &IO, SymbolLanguage &Value)