23#include "llvm/ADT/STLExtras.h"
24#include "llvm/ADT/StringMap.h"
25#include "llvm/Support/Error.h"
26#include "llvm/Support/Path.h"
41 static const llvm::StringMap<CommentKind> KindMap = {
56 auto It = KindMap.find(KindStr);
57 if (It != KindMap.end()) {
68 return "ParagraphComment";
72 return "InlineCommandComment";
74 return "HTMLStartTagComment";
76 return "HTMLEndTagComment";
78 return "BlockCommandComment";
80 return "ParamCommandComment";
82 return "TParamCommandComment";
84 return "VerbatimBlockComment";
86 return "VerbatimBlockLineComment";
88 return "VerbatimLineComment";
92 llvm_unreachable(
"Unhandled CommentKind");
99 if (Values.empty() || !Values[0])
100 return llvm::createStringError(llvm::inconvertibleErrorCode(),
101 "no value to reduce");
103 for (
auto &I : Values)
104 Merged->merge(std::move(*
static_cast<T *
>(I)));
110 while (!ChildrenToMerge.empty()) {
111 T *Ptr = ChildrenToMerge.front().Ptr;
112 ChildrenToMerge.pop_front();
114 auto It = llvm::find_if(
115 Children, [Ptr](
const auto &C) {
return C.Ptr->USR == Ptr->USR; });
117 if (It == Children.end()) {
119 NewNode->Ptr->merge(std::move(*Ptr));
120 Children.push_back(*NewNode);
122 It->Ptr->merge(std::move(*Ptr));
130 while (!ChildrenToMerge.empty()) {
131 Reference *Ptr = ChildrenToMerge.front().Ptr;
132 ChildrenToMerge.pop_front();
134 auto It = llvm::find_if(
135 Children, [Ptr](
const auto &C) {
return C.Ptr->USR == Ptr->
USR; });
136 if (It == Children.end()) {
138 NewNode->Ptr->USR = Ptr->
USR;
139 NewNode->Ptr->RefType = Ptr->
RefType;
140 NewNode->Ptr->merge(std::move(*Ptr));
141 Children.push_back(*NewNode);
143 It->Ptr->merge(std::move(*Ptr));
150 while (!Source.empty()) {
151 T *Ptr = Source.front().Ptr;
154 if (!llvm::any_of(Target,
155 [Ptr](
const auto &E) {
return *E.Ptr == *Ptr; })) {
164 while (!Source.empty()) {
168 if (llvm::none_of(Target,
169 [Ptr](
const auto &E) {
return *E.Ptr == *Ptr; })) {
177 llvm::BumpPtrAllocator &Arena) {
179 switch (NewInfo->
IT) {
205 return llvm::createStringError(llvm::inconvertibleErrorCode(),
206 "unknown info type");
210 if (Reduced->
IT != NewInfo->
IT)
211 return llvm::createStringError(llvm::inconvertibleErrorCode(),
212 "info types mismatch");
214 switch (Reduced->
IT) {
221 std::move(*
static_cast<RecordInfo *
>(NewInfo)));
224 static_cast<EnumInfo *
>(Reduced)->merge(
225 std::move(*
static_cast<EnumInfo *
>(NewInfo)));
240 static_cast<VarInfo *
>(Reduced)->merge(
241 std::move(*
static_cast<VarInfo *
>(NewInfo)));
245 std::move(*
static_cast<FriendInfo *
>(NewInfo)));
248 return llvm::createStringError(llvm::inconvertibleErrorCode(),
249 "unknown info type");
252 return llvm::Error::success();
257 if (Values.empty() || !Values[0])
258 return llvm::createStringError(llvm::inconvertibleErrorCode(),
259 "no info values to merge");
261 switch (Values[0]->IT) {
279 return llvm::createStringError(llvm::inconvertibleErrorCode(),
280 "unexpected info type");
282 llvm_unreachable(
"unhandled enumerator");
292 llvm::BumpPtrAllocator &Arena) {
321 if (FirstCI < SecondCI)
324 if (FirstCI == SecondCI) {
334 llvm::BumpPtrAllocator &Arena) {
348 for (
size_t Idx = 0; Idx < Other.
Children.size(); ++Idx) {
355static llvm::SmallString<64>
357 const StringRef &Name,
const StringRef &CurrentPath) {
358 llvm::SmallString<64> FilePath;
360 if (CurrentPath != Path) {
362 for (llvm::sys::path::const_iterator I =
363 llvm::sys::path::begin(CurrentPath);
364 I != llvm::sys::path::end(CurrentPath); ++I)
365 llvm::sys::path::append(FilePath,
"..");
366 llvm::sys::path::append(FilePath, Path);
372 llvm::sys::path::append(FilePath, Name);
374 return llvm::sys::path::relative_path(FilePath);
408 "Merging reference with default InfoType");
425 Ref.merge(std::move(Other.Ref));
426 SymbolInfo::merge(std::move(Other));
436 if (!Other.
Params.empty())
463 if (
Namespace.empty() && !Other.Namespace.empty())
480 if (!Other.
Loc.empty()) {
481 for (const auto &L : Other.Loc) {
482 Location *NewL = allocatePtr<Location>(Arena, L);
483 Loc.push_back(*allocateListNode<Location>(Arena, NewL));
491 DefLoc = std::move(Other.DefLoc);
536 llvm::BumpPtrAllocator &Arena)
539 for (const auto &Desc : Other.Description) {
540 CommentInfo *NewDesc = allocatePtr<CommentInfo>(Arena, Desc, Arena);
541 Description.push_back(*allocateListNode<CommentInfo>(Arena, NewDesc));
548 if (!llvm::to_underlying(
TagType))
551 if (
Members.empty() && !Other.Members.empty())
553 if (
Bases.empty() && !Other.Bases.empty())
555 if (
Parents.empty() && !Other.Parents.empty())
559 if (
Friends.empty() && !Other.Friends.empty())
568 SymbolInfo::merge(std::move(Other));
572 llvm::BumpPtrAllocator &Arena)
575 for (const auto &Desc : Other.Description) {
576 CommentInfo *NewDesc = allocatePtr<CommentInfo>(Arena, Desc, Arena);
577 Description.push_back(*allocateListNode<CommentInfo>(Arena, NewDesc));
587 BaseType = std::move(Other.BaseType);
588 if (
Members.empty() && !Other.Members.empty())
590 SymbolInfo::merge(std::move(Other));
602 Parent = std::move(Other.Parent);
603 if (
Params.empty() && !Other.Params.empty())
607 SymbolInfo::merge(std::move(Other));
618 SymbolInfo::merge(std::move(Other));
627 if (
Template.Constraints.empty() && !Other.Template.Constraints.empty())
630 if (
Template.Params.empty() && !Other.Template.Params.empty())
632 SymbolInfo::merge(std::move(Other));
640 Type = std::move(Other.Type);
641 SymbolInfo::merge(std::move(Other));
647 llvm::BumpPtrAllocator &Arena)
668 return "@GlobalNamespace";
672 return "GlobalNamespace";
674 return internString(
"@nonymous_record_" + toHex(llvm::toStringRef(
USR)));
676 return internString(
"@nonymous_enum_" + toHex(llvm::toStringRef(
USR)));
678 return internString(
"@nonymous_typedef_" + toHex(llvm::toStringRef(
USR)));
680 return internString(
"@nonymous_function_" + toHex(llvm::toStringRef(
USR)));
682 return internString(
"@nonymous_concept_" + toHex(llvm::toStringRef(
USR)));
684 return internString(
"@nonymous_variable_" + toHex(llvm::toStringRef(
USR)));
686 return internString(
"@nonymous_friend_" + toHex(llvm::toStringRef(
USR)));
690 llvm_unreachable(
"Invalid InfoType.");
699 int Cmp =
Name.compare_insensitive(Other.
Name);
709 std::vector<const Index *> SortedChildren;
710 SortedChildren.reserve(
Children.size());
712 SortedChildren.push_back(&C);
713 llvm::sort(SortedChildren,
714 [](
const Index *A,
const Index *B) {
return *A < *B; });
715 return SortedChildren;
729 clang::DiagnosticsEngine &
Diags,
735 llvm::SmallString<128> SourceRootDir(
SourceRoot);
738 llvm::sys::fs::current_path(SourceRootDir);
739 this->SourceRoot = std::string(SourceRootDir);
744 this->RepositoryUrl->insert(0,
"https://");
T * allocatePtr(llvm::BumpPtrAllocator &Alloc, Args &&...args)
void reduceChildren< Reference >(DocList< Reference > &Children, DocList< Reference > &&ChildrenToMerge)
static void mergeUnkeyed(DocList< T > &Target, DocList< T > &&Source)
@ CK_InlineCommandComment
@ CK_VerbatimBlockLineComment
@ CK_VerbatimBlockComment
@ CK_TParamCommandComment
static llvm::SmallString< 64 > calculateRelativeFilePath(const InfoType &Type, const StringRef &Path, const StringRef &Name, const StringRef &CurrentPath)
llvm::simple_ilist< InfoNode< T > > DocList
thread_local llvm::BumpPtrAllocator PersistentArena
T * allocateTransient(Args &&...args)
void mergeUnkeyed< CommentInfo >(DocList< CommentInfo > &Target, DocList< CommentInfo > &&Source)
ConcurrentStringPool & getGlobalStringPool()
CommentKind stringToCommentKind(llvm::StringRef KindStr)
llvm::ArrayRef< T > deepCopyArray(llvm::ArrayRef< T > V, llvm::BumpPtrAllocator &Alloc)
InfoNode< T > * allocateListNodePersistent(Args &&...args)
InfoNode< T > * allocateListNode(llvm::BumpPtrAllocator &Alloc, Args &&...args)
thread_local llvm::BumpPtrAllocator TransientArena
StringRef internString(const Twine &T)
llvm::ArrayRef< T > allocateArray(llvm::SmallVectorImpl< T > &V, llvm::BumpPtrAllocator &Alloc)
llvm::Expected< Info * > mergeInfos(SmallVectorImpl< Info * > &Values)
static void reduceChildren(DocList< T > &Children, DocList< T > &&ChildrenToMerge)
static llvm::Expected< Info * > reduce(SmallVectorImpl< Info * > &Values)
std::array< uint8_t, 20 > SymbolID
llvm::StringRef commentKindToString(CommentKind Kind)
llvm::Error mergeSingleInfo(doc::Info *&Reduced, doc::Info *NewInfo, llvm::BumpPtrAllocator &Arena)
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
std::optional< std::string > RepositoryUrl
std::vector< std::string > UserStylesheets
ClangDocContext(tooling::ExecutionContext *ECtx, StringRef ProjectName, bool PublicOnly, StringRef OutDirectory, StringRef SourceRoot, StringRef RepositoryUrl, StringRef RepositoryCodeLinePrefix, StringRef Base, std::vector< std::string > UserStylesheets, clang::DiagnosticsEngine &Diags, OutputFormatTy Format, bool FTimeTrace=false)
tooling::ExecutionContext * ECtx
clang::DiagnosticsEngine & Diags
std::optional< std::string > RepositoryLinePrefix
void merge(ConceptInfo &&I)
StringRef ConstraintExpression
llvm::ArrayRef< EnumValueInfo > Members
std::optional< TypeInfo > BaseType
DocList< CommentInfo > Description
Comment description of this field.
EnumValueInfo(StringRef Name=StringRef(), StringRef Value=StringRef("0"), StringRef ValueExpr=StringRef())
std::optional< TypeInfo > ReturnType
void merge(FriendInfo &&Other)
llvm::ArrayRef< FieldTypeInfo > Params
std::optional< TemplateInfo > Template
bool mergeable(const FriendInfo &Other)
FunctionInfo(SymbolID USR=SymbolID())
llvm::ArrayRef< FieldTypeInfo > Params
void merge(FunctionInfo &&I)
std::optional< TemplateInfo > Template
bool operator<(const Index &Other) const
llvm::StringMap< Index > Children
std::vector< const Index * > getSortedChildren() const
StringRef getRelativeFilePath(const StringRef &CurrentPath) const
Returns the file path for this Info relative to CurrentPath.
Info(InfoType IT=InfoType::IT_default, SymbolID USR=SymbolID(), StringRef Name=StringRef(), StringRef Path=StringRef())
bool mergeable(const Info &Other)
DocList< CommentInfo > Description
llvm::ArrayRef< Reference > Namespace
StringRef extractName() const
StringRef DocumentationFileName
StringRef getFileBaseName() const
Returns the basename that should be used for this Info.
DocList< CommentInfo > Description
NamespaceInfo(SymbolID USR=SymbolID(), StringRef Name=StringRef(), StringRef Path=StringRef())
void merge(NamespaceInfo &&I)
llvm::ArrayRef< BaseRecordInfo > Bases
llvm::ArrayRef< FriendInfo > Friends
RecordInfo(SymbolID USR=SymbolID(), StringRef Name=StringRef(), StringRef Path=StringRef())
llvm::ArrayRef< Reference > VirtualParents
std::optional< TemplateInfo > Template
llvm::ArrayRef< Reference > Parents
llvm::ArrayRef< MemberTypeInfo > Members
void merge(RecordInfo &&I)
void merge(Reference &&I)
StringRef DocumentationFileName
Reference(SymbolID USR=SymbolID(), StringRef Name=StringRef(), InfoType IT=InfoType::IT_default)
bool mergeable(const Reference &Other)
StringRef getFileBaseName() const
Returns the basename that should be used for this Reference.
StringRef getRelativeFilePath(const StringRef &CurrentPath) const
Returns the path for this Reference relative to CurrentPath.
DocList< Reference > Records
DocList< EnumInfo > Enums
DocList< ConceptInfo > Concepts
DocList< VarInfo > Variables
DocList< FunctionInfo > Functions
DocList< Reference > Namespaces
DocList< TypedefInfo > Typedefs
SymbolInfo(InfoType IT, SymbolID USR=SymbolID(), StringRef Name=StringRef(), StringRef Path=StringRef())
std::optional< Location > DefLoc
llvm::ArrayRef< TemplateParamInfo > Params
llvm::ArrayRef< ConstraintInfo > Constraints
std::optional< TemplateSpecializationInfo > Specialization
SymbolID SpecializationOf
TemplateSpecializationInfo()=default
llvm::ArrayRef< TemplateParamInfo > Params
void merge(TypedefInfo &&I)
std::optional< TemplateInfo > Template
TypedefInfo(SymbolID USR=SymbolID())