This is a set of squashed reverts of recent clang doc patches, since its breaking something on Darwin builders: https://lab.llvm.org/buildbot/#/builders/23/builds/19172 Revert "[clang-doc][nfc] Default initialize all StringRef members (#191641)" This reverts commit155b9b354c. Revert "[clang-doc] Initialize StringRef members in Info types (#191637)" This reverts commit489dab3827. Revert "[clang-doc] Initialize member variable (#191570)" This reverts commit5d64a44a84. Revert "[clang-doc] Merge data into persistent memory (#190056)" This reverts commit21e0034c69. Revert "[clang-doc] Support deep copy between arenas for merging (#190055)" This reverts commitc70dae8b0c.
541 lines
19 KiB
C++
541 lines
19 KiB
C++
///===-- Representation.cpp - ClangDoc Representation -----------*- 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines the merging of different types of infos. The data in the
|
|
// calling Info is preserved during a merge unless that field is empty or
|
|
// default. In that case, the data from the parameter Info is used to replace
|
|
// the empty or default data.
|
|
//
|
|
// For most fields, the first decl seen provides the data. Exceptions to this
|
|
// include the location and description fields, which are collections of data on
|
|
// all decls related to a given definition. All other fields are ignored in new
|
|
// decls unless the first seen decl didn't, for whatever reason, incorporate
|
|
// data on that field (e.g. a forward declared class wouldn't have information
|
|
// on members on the forward declaration, but would have the class name).
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
#include "Representation.h"
|
|
#include "llvm/ADT/StringMap.h"
|
|
#include "llvm/Support/Error.h"
|
|
#include "llvm/Support/Path.h"
|
|
|
|
namespace clang {
|
|
namespace doc {
|
|
|
|
// Thread local arenas usable in each thread pool
|
|
thread_local llvm::BumpPtrAllocator TransientArena;
|
|
thread_local llvm::BumpPtrAllocator PersistentArena;
|
|
|
|
ConcurrentStringPool &getGlobalStringPool() {
|
|
static ConcurrentStringPool GlobalPool;
|
|
return GlobalPool;
|
|
}
|
|
|
|
CommentKind stringToCommentKind(llvm::StringRef KindStr) {
|
|
static const llvm::StringMap<CommentKind> KindMap = {
|
|
{"FullComment", CommentKind::CK_FullComment},
|
|
{"ParagraphComment", CommentKind::CK_ParagraphComment},
|
|
{"TextComment", CommentKind::CK_TextComment},
|
|
{"InlineCommandComment", CommentKind::CK_InlineCommandComment},
|
|
{"HTMLStartTagComment", CommentKind::CK_HTMLStartTagComment},
|
|
{"HTMLEndTagComment", CommentKind::CK_HTMLEndTagComment},
|
|
{"BlockCommandComment", CommentKind::CK_BlockCommandComment},
|
|
{"ParamCommandComment", CommentKind::CK_ParamCommandComment},
|
|
{"TParamCommandComment", CommentKind::CK_TParamCommandComment},
|
|
{"VerbatimBlockComment", CommentKind::CK_VerbatimBlockComment},
|
|
{"VerbatimBlockLineComment", CommentKind::CK_VerbatimBlockLineComment},
|
|
{"VerbatimLineComment", CommentKind::CK_VerbatimLineComment},
|
|
};
|
|
|
|
auto It = KindMap.find(KindStr);
|
|
if (It != KindMap.end()) {
|
|
return It->second;
|
|
}
|
|
return CommentKind::CK_Unknown;
|
|
}
|
|
|
|
llvm::StringRef commentKindToString(CommentKind Kind) {
|
|
switch (Kind) {
|
|
case CommentKind::CK_FullComment:
|
|
return "FullComment";
|
|
case CommentKind::CK_ParagraphComment:
|
|
return "ParagraphComment";
|
|
case CommentKind::CK_TextComment:
|
|
return "TextComment";
|
|
case CommentKind::CK_InlineCommandComment:
|
|
return "InlineCommandComment";
|
|
case CommentKind::CK_HTMLStartTagComment:
|
|
return "HTMLStartTagComment";
|
|
case CommentKind::CK_HTMLEndTagComment:
|
|
return "HTMLEndTagComment";
|
|
case CommentKind::CK_BlockCommandComment:
|
|
return "BlockCommandComment";
|
|
case CommentKind::CK_ParamCommandComment:
|
|
return "ParamCommandComment";
|
|
case CommentKind::CK_TParamCommandComment:
|
|
return "TParamCommandComment";
|
|
case CommentKind::CK_VerbatimBlockComment:
|
|
return "VerbatimBlockComment";
|
|
case CommentKind::CK_VerbatimBlockLineComment:
|
|
return "VerbatimBlockLineComment";
|
|
case CommentKind::CK_VerbatimLineComment:
|
|
return "VerbatimLineComment";
|
|
case CommentKind::CK_Unknown:
|
|
return "Unknown";
|
|
}
|
|
llvm_unreachable("Unhandled CommentKind");
|
|
}
|
|
|
|
const SymbolID EmptySID = SymbolID();
|
|
|
|
template <typename T>
|
|
static llvm::Expected<OwnedPtr<Info>> reduce(OwningPtrArray<Info> &Values) {
|
|
if (Values.empty() || !Values[0])
|
|
return llvm::createStringError(llvm::inconvertibleErrorCode(),
|
|
"no value to reduce");
|
|
OwnedPtr<Info> Merged = allocatePtr<T>(Values[0]->USR);
|
|
T *Tmp = static_cast<T *>(getPtr(Merged));
|
|
for (auto &I : Values)
|
|
Tmp->merge(std::move(*static_cast<T *>(getPtr(I))));
|
|
return std::move(Merged);
|
|
}
|
|
|
|
// Return the index of the matching child in the vector, or -1 if merge is not
|
|
// necessary.
|
|
template <typename T>
|
|
static int getChildIndexIfExists(OwningVec<T> &Children, T &ChildToMerge) {
|
|
for (unsigned long I = 0; I < Children.size(); I++) {
|
|
if (ChildToMerge.USR == Children[I].USR)
|
|
return I;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
template <typename T>
|
|
static void reduceChildren(llvm::simple_ilist<T> &Children,
|
|
llvm::simple_ilist<T> &&ChildrenToMerge) {
|
|
while (!ChildrenToMerge.empty()) {
|
|
T *ChildToMerge = &ChildrenToMerge.front();
|
|
ChildrenToMerge.pop_front();
|
|
|
|
auto It = llvm::find_if(
|
|
Children, [&](const T &C) { return C.USR == ChildToMerge->USR; });
|
|
if (It == Children.end()) {
|
|
Children.push_back(*ChildToMerge);
|
|
} else {
|
|
It->merge(std::move(*ChildToMerge));
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
static void reduceChildren(OwningVec<T> &Children,
|
|
OwningVec<T> &&ChildrenToMerge) {
|
|
for (auto &ChildToMerge : ChildrenToMerge) {
|
|
int MergeIdx = getChildIndexIfExists(Children, ChildToMerge);
|
|
if (MergeIdx == -1) {
|
|
Children.push_back(std::move(ChildToMerge));
|
|
continue;
|
|
}
|
|
Children[MergeIdx].merge(std::move(ChildToMerge));
|
|
}
|
|
}
|
|
|
|
template <typename Container>
|
|
static void mergeUnkeyed(Container &Target, Container &&Source) {
|
|
for (auto &Item : Source) {
|
|
if (llvm::none_of(Target, [&](const auto &E) { return E == Item; }))
|
|
Target.push_back(std::move(Item));
|
|
}
|
|
}
|
|
|
|
// Dispatch function.
|
|
llvm::Expected<OwnedPtr<Info>> mergeInfos(OwningPtrArray<Info> &Values) {
|
|
if (Values.empty() || !Values[0])
|
|
return llvm::createStringError(llvm::inconvertibleErrorCode(),
|
|
"no info values to merge");
|
|
|
|
switch (Values[0]->IT) {
|
|
case InfoType::IT_namespace:
|
|
return reduce<NamespaceInfo>(Values);
|
|
case InfoType::IT_record:
|
|
return reduce<RecordInfo>(Values);
|
|
case InfoType::IT_enum:
|
|
return reduce<EnumInfo>(Values);
|
|
case InfoType::IT_function:
|
|
return reduce<FunctionInfo>(Values);
|
|
case InfoType::IT_typedef:
|
|
return reduce<TypedefInfo>(Values);
|
|
case InfoType::IT_concept:
|
|
return reduce<ConceptInfo>(Values);
|
|
case InfoType::IT_variable:
|
|
return reduce<VarInfo>(Values);
|
|
case InfoType::IT_friend:
|
|
return reduce<FriendInfo>(Values);
|
|
case InfoType::IT_default:
|
|
return llvm::createStringError(llvm::inconvertibleErrorCode(),
|
|
"unexpected info type");
|
|
}
|
|
llvm_unreachable("unhandled enumerator");
|
|
}
|
|
|
|
bool CommentInfo::operator==(const CommentInfo &Other) const {
|
|
auto FirstCI = std::tie(Kind, Text, Name, Direction, ParamName, CloseName,
|
|
SelfClosing, Explicit, AttrKeys, AttrValues, Args);
|
|
auto SecondCI =
|
|
std::tie(Other.Kind, Other.Text, Other.Name, Other.Direction,
|
|
Other.ParamName, Other.CloseName, Other.SelfClosing,
|
|
Other.Explicit, Other.AttrKeys, Other.AttrValues, Other.Args);
|
|
|
|
if (FirstCI != SecondCI || Children.size() != Other.Children.size())
|
|
return false;
|
|
|
|
return std::equal(Children.begin(), Children.end(), Other.Children.begin(),
|
|
Other.Children.end());
|
|
}
|
|
|
|
bool CommentInfo::operator<(const CommentInfo &Other) const {
|
|
auto FirstCI = std::tie(Kind, Text, Name, Direction, ParamName, CloseName,
|
|
SelfClosing, Explicit, AttrKeys, AttrValues, Args);
|
|
auto SecondCI =
|
|
std::tie(Other.Kind, Other.Text, Other.Name, Other.Direction,
|
|
Other.ParamName, Other.CloseName, Other.SelfClosing,
|
|
Other.Explicit, Other.AttrKeys, Other.AttrValues, Other.Args);
|
|
|
|
if (FirstCI < SecondCI)
|
|
return true;
|
|
|
|
if (FirstCI == SecondCI) {
|
|
return std::lexicographical_compare(Children.begin(), Children.end(),
|
|
Other.Children.begin(),
|
|
Other.Children.end());
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static llvm::SmallString<64>
|
|
calculateRelativeFilePath(const InfoType &Type, const StringRef &Path,
|
|
const StringRef &Name, const StringRef &CurrentPath) {
|
|
llvm::SmallString<64> FilePath;
|
|
|
|
if (CurrentPath != Path) {
|
|
// iterate back to the top
|
|
for (llvm::sys::path::const_iterator I =
|
|
llvm::sys::path::begin(CurrentPath);
|
|
I != llvm::sys::path::end(CurrentPath); ++I)
|
|
llvm::sys::path::append(FilePath, "..");
|
|
llvm::sys::path::append(FilePath, Path);
|
|
}
|
|
|
|
// Namespace references have a Path to the parent namespace, but
|
|
// the file is actually in the subdirectory for the namespace.
|
|
if (Type == doc::InfoType::IT_namespace)
|
|
llvm::sys::path::append(FilePath, Name);
|
|
|
|
return llvm::sys::path::relative_path(FilePath);
|
|
}
|
|
|
|
StringRef Reference::getRelativeFilePath(const StringRef &CurrentPath) const {
|
|
return internString(
|
|
calculateRelativeFilePath(RefType, Path, Name, CurrentPath));
|
|
}
|
|
|
|
StringRef Reference::getFileBaseName() const {
|
|
if (RefType == InfoType::IT_namespace)
|
|
return "index";
|
|
|
|
return Name;
|
|
}
|
|
|
|
StringRef Info::getRelativeFilePath(const StringRef &CurrentPath) const {
|
|
return internString(
|
|
calculateRelativeFilePath(IT, Path, extractName(), CurrentPath));
|
|
}
|
|
|
|
StringRef Info::getFileBaseName() const {
|
|
if (IT == InfoType::IT_namespace)
|
|
return "index";
|
|
|
|
return extractName();
|
|
}
|
|
|
|
bool Reference::mergeable(const Reference &Other) {
|
|
return RefType == Other.RefType && USR == Other.USR;
|
|
}
|
|
|
|
void Reference::merge(Reference &&Other) {
|
|
assert(mergeable(Other));
|
|
if (Name.empty())
|
|
Name = Other.Name;
|
|
if (Path.empty())
|
|
Path = Other.Path;
|
|
if (DocumentationFileName.empty())
|
|
DocumentationFileName = Other.DocumentationFileName;
|
|
}
|
|
|
|
bool FriendInfo::mergeable(const FriendInfo &Other) {
|
|
return Ref.USR == Other.Ref.USR && Ref.Name == Other.Ref.Name;
|
|
}
|
|
|
|
void FriendInfo::merge(FriendInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
Ref.merge(std::move(Other.Ref));
|
|
SymbolInfo::merge(std::move(Other));
|
|
}
|
|
|
|
void Info::mergeBase(Info &&Other) {
|
|
assert(mergeable(Other));
|
|
if (USR == EmptySID)
|
|
USR = Other.USR;
|
|
if (Name == "")
|
|
Name = Other.Name;
|
|
if (Path == "")
|
|
Path = Other.Path;
|
|
if (Namespace.empty())
|
|
Namespace = std::move(Other.Namespace);
|
|
// Unconditionally extend the description, since each decl may have a comment.
|
|
mergeUnkeyed(Description, std::move(Other.Description));
|
|
if (ParentUSR == EmptySID)
|
|
ParentUSR = Other.ParentUSR;
|
|
if (DocumentationFileName.empty())
|
|
DocumentationFileName = Other.DocumentationFileName;
|
|
}
|
|
|
|
bool Info::mergeable(const Info &Other) {
|
|
return IT == Other.IT && USR == Other.USR;
|
|
}
|
|
|
|
void SymbolInfo::merge(SymbolInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!DefLoc)
|
|
DefLoc = std::move(Other.DefLoc);
|
|
// Unconditionally extend the list of locations, since we want all of them.
|
|
mergeUnkeyed(Loc, std::move(Other.Loc));
|
|
mergeBase(std::move(Other));
|
|
if (MangledName.empty())
|
|
MangledName = std::move(Other.MangledName);
|
|
}
|
|
|
|
NamespaceInfo::NamespaceInfo(SymbolID USR, StringRef Name, StringRef Path)
|
|
: Info(InfoType::IT_namespace, USR, Name, Path) {}
|
|
|
|
void NamespaceInfo::merge(NamespaceInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
// Reduce children if necessary.
|
|
reduceChildren(Children.Namespaces, std::move(Other.Children.Namespaces));
|
|
reduceChildren(Children.Records, std::move(Other.Children.Records));
|
|
reduceChildren(Children.Functions, std::move(Other.Children.Functions));
|
|
reduceChildren(Children.Enums, std::move(Other.Children.Enums));
|
|
reduceChildren(Children.Typedefs, std::move(Other.Children.Typedefs));
|
|
reduceChildren(Children.Concepts, std::move(Other.Children.Concepts));
|
|
reduceChildren(Children.Variables, std::move(Other.Children.Variables));
|
|
mergeBase(std::move(Other));
|
|
}
|
|
|
|
RecordInfo::RecordInfo(SymbolID USR, StringRef Name, StringRef Path)
|
|
: SymbolInfo(InfoType::IT_record, USR, Name, Path) {}
|
|
|
|
void RecordInfo::merge(RecordInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!llvm::to_underlying(TagType))
|
|
TagType = Other.TagType;
|
|
IsTypeDef = IsTypeDef || Other.IsTypeDef;
|
|
if (Members.empty())
|
|
Members = std::move(Other.Members);
|
|
if (Bases.empty())
|
|
Bases = std::move(Other.Bases);
|
|
if (Parents.empty())
|
|
Parents = std::move(Other.Parents);
|
|
if (VirtualParents.empty())
|
|
VirtualParents = std::move(Other.VirtualParents);
|
|
if (Friends.empty())
|
|
Friends = std::move(Other.Friends);
|
|
// Reduce children if necessary.
|
|
reduceChildren(Children.Records, std::move(Other.Children.Records));
|
|
reduceChildren(Children.Functions, std::move(Other.Children.Functions));
|
|
reduceChildren(Children.Enums, std::move(Other.Children.Enums));
|
|
reduceChildren(Children.Typedefs, std::move(Other.Children.Typedefs));
|
|
SymbolInfo::merge(std::move(Other));
|
|
if (!Template)
|
|
Template = Other.Template;
|
|
}
|
|
|
|
void EnumInfo::merge(EnumInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!Scoped)
|
|
Scoped = Other.Scoped;
|
|
if (Members.empty())
|
|
Members = std::move(Other.Members);
|
|
SymbolInfo::merge(std::move(Other));
|
|
}
|
|
|
|
void FunctionInfo::merge(FunctionInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!IsMethod)
|
|
IsMethod = Other.IsMethod;
|
|
if (!Access)
|
|
Access = Other.Access;
|
|
if (ReturnType.Type.USR == EmptySID && ReturnType.Type.Name == "")
|
|
ReturnType = std::move(Other.ReturnType);
|
|
if (Parent.USR == EmptySID && Parent.Name == "")
|
|
Parent = std::move(Other.Parent);
|
|
if (Params.empty())
|
|
Params = std::move(Other.Params);
|
|
SymbolInfo::merge(std::move(Other));
|
|
if (!Template)
|
|
Template = Other.Template;
|
|
}
|
|
|
|
void TypedefInfo::merge(TypedefInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!IsUsing)
|
|
IsUsing = Other.IsUsing;
|
|
if (Underlying.Type.Name == "")
|
|
Underlying = Other.Underlying;
|
|
if (!Template)
|
|
Template = Other.Template;
|
|
SymbolInfo::merge(std::move(Other));
|
|
}
|
|
|
|
void ConceptInfo::merge(ConceptInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!IsType)
|
|
IsType = Other.IsType;
|
|
if (ConstraintExpression.empty())
|
|
ConstraintExpression = std::move(Other.ConstraintExpression);
|
|
if (Template.Constraints.empty())
|
|
Template.Constraints = std::move(Other.Template.Constraints);
|
|
if (Template.Params.empty())
|
|
Template.Params = std::move(Other.Template.Params);
|
|
SymbolInfo::merge(std::move(Other));
|
|
}
|
|
|
|
void VarInfo::merge(VarInfo &&Other) {
|
|
assert(mergeable(Other));
|
|
if (!IsStatic)
|
|
IsStatic = Other.IsStatic;
|
|
if (Type.Type.USR == EmptySID && Type.Type.Name == "")
|
|
Type = std::move(Other.Type);
|
|
SymbolInfo::merge(std::move(Other));
|
|
}
|
|
|
|
BaseRecordInfo::BaseRecordInfo() : RecordInfo() {}
|
|
|
|
BaseRecordInfo::BaseRecordInfo(SymbolID USR, StringRef Name, StringRef Path,
|
|
bool IsVirtual, AccessSpecifier Access,
|
|
bool IsParent)
|
|
: RecordInfo(USR, Name, Path), Access(Access), IsVirtual(IsVirtual),
|
|
IsParent(IsParent) {}
|
|
|
|
StringRef Info::extractName() const {
|
|
if (!Name.empty())
|
|
return Name;
|
|
|
|
switch (IT) {
|
|
case InfoType::IT_namespace:
|
|
// Cover the case where the project contains a base namespace called
|
|
// 'GlobalNamespace' (i.e. a namespace at the same level as the global
|
|
// namespace, which would conflict with the hard-coded global namespace name
|
|
// below.)
|
|
if (Name == "GlobalNamespace" && Namespace.empty())
|
|
return "@GlobalNamespace";
|
|
// The case of anonymous namespaces is taken care of in serialization,
|
|
// so here we can safely assume an unnamed namespace is the global
|
|
// one.
|
|
return "GlobalNamespace";
|
|
case InfoType::IT_record:
|
|
return internString("@nonymous_record_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_enum:
|
|
return internString("@nonymous_enum_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_typedef:
|
|
return internString("@nonymous_typedef_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_function:
|
|
return internString("@nonymous_function_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_concept:
|
|
return internString("@nonymous_concept_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_variable:
|
|
return internString("@nonymous_variable_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_friend:
|
|
return internString("@nonymous_friend_" + toHex(llvm::toStringRef(USR)));
|
|
case InfoType::IT_default:
|
|
return internString("@nonymous_" + toHex(llvm::toStringRef(USR)));
|
|
}
|
|
llvm_unreachable("Invalid InfoType.");
|
|
return "";
|
|
}
|
|
|
|
// Order is based on the Name attribute: case insensitive order
|
|
bool Index::operator<(const Index &Other) const {
|
|
// Start with case-insensitive (e.g., 'apple' < 'Zebra').
|
|
// This prevents 'Zebra' from appearing before 'apple' due to ASCII values,
|
|
// where uppercase letters have a lower numeric value than lowercase.
|
|
int Cmp = Name.compare_insensitive(Other.Name);
|
|
if (Cmp != 0)
|
|
return Cmp < 0;
|
|
|
|
// If names are identical, we fall back to standard string comparison where
|
|
// uppercase precedes lowercase (e.g., 'Apple' < 'apple').
|
|
return Name < Other.Name;
|
|
}
|
|
|
|
OwningVec<const Index *> Index::getSortedChildren() const {
|
|
OwningVec<const Index *> SortedChildren;
|
|
SortedChildren.reserve(Children.size());
|
|
for (const auto &[_, C] : Children)
|
|
SortedChildren.push_back(&C);
|
|
llvm::sort(SortedChildren,
|
|
[](const Index *A, const Index *B) { return *A < *B; });
|
|
return SortedChildren;
|
|
}
|
|
|
|
void Index::sort() {
|
|
for (auto &[_, C] : Children)
|
|
C.sort();
|
|
}
|
|
|
|
ClangDocContext::ClangDocContext(tooling::ExecutionContext *ECtx,
|
|
StringRef ProjectName, bool PublicOnly,
|
|
StringRef OutDirectory, StringRef SourceRoot,
|
|
StringRef RepositoryUrl,
|
|
StringRef RepositoryLinePrefix, StringRef Base,
|
|
std::vector<std::string> UserStylesheets,
|
|
clang::DiagnosticsEngine &Diags,
|
|
OutputFormatTy Format, bool FTimeTrace)
|
|
: ECtx(ECtx), ProjectName(ProjectName), OutDirectory(OutDirectory),
|
|
SourceRoot(std::string(SourceRoot)), UserStylesheets(UserStylesheets),
|
|
Base(Base), Diags(Diags), Format(Format), PublicOnly(PublicOnly),
|
|
FTimeTrace(FTimeTrace) {
|
|
llvm::SmallString<128> SourceRootDir(SourceRoot);
|
|
if (SourceRoot.empty())
|
|
// If no SourceRoot was provided the current path is used as the default
|
|
llvm::sys::fs::current_path(SourceRootDir);
|
|
this->SourceRoot = std::string(SourceRootDir);
|
|
if (!RepositoryUrl.empty()) {
|
|
this->RepositoryUrl = std::string(RepositoryUrl);
|
|
if (!RepositoryUrl.empty() && !RepositoryUrl.starts_with("http://") &&
|
|
!RepositoryUrl.starts_with("https://"))
|
|
this->RepositoryUrl->insert(0, "https://");
|
|
|
|
if (!RepositoryLinePrefix.empty())
|
|
this->RepositoryLinePrefix = std::string(RepositoryLinePrefix);
|
|
}
|
|
}
|
|
|
|
void ScopeChildren::sort() {
|
|
Namespaces.sort();
|
|
llvm::sort(Records);
|
|
llvm::sort(Functions);
|
|
llvm::sort(Enums);
|
|
llvm::sort(Typedefs);
|
|
llvm::sort(Concepts);
|
|
llvm::sort(Variables);
|
|
}
|
|
} // namespace doc
|
|
} // namespace clang
|