2018-09-11 23:56:55 +08:00
|
|
|
//===-- MDGenerator.cpp - Markdown Generator --------------------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2018-09-11 23:56:55 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Generators.h"
|
|
|
|
#include "Representation.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/Path.h"
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace doc {
|
|
|
|
|
|
|
|
// Markdown generation
|
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static std::string genItalic(const Twine &Text) {
|
|
|
|
return "*" + Text.str() + "*";
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static std::string genEmphasis(const Twine &Text) {
|
|
|
|
return "**" + Text.str() + "**";
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
|
2019-07-13 02:32:00 +08:00
|
|
|
static std::string
|
|
|
|
genReferenceList(const llvm::SmallVectorImpl<Reference> &Refs) {
|
|
|
|
std::string Buffer;
|
|
|
|
llvm::raw_string_ostream Stream(Buffer);
|
|
|
|
for (const auto &R : Refs) {
|
|
|
|
if (&R != Refs.begin())
|
|
|
|
Stream << ", ";
|
|
|
|
Stream << R.Name;
|
|
|
|
}
|
|
|
|
return Stream.str();
|
|
|
|
}
|
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static void writeLine(const Twine &Text, raw_ostream &OS) {
|
|
|
|
OS << Text << "\n\n";
|
|
|
|
}
|
2018-09-11 23:56:55 +08:00
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static void writeNewLine(raw_ostream &OS) { OS << "\n\n"; }
|
2018-09-11 23:56:55 +08:00
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static void writeHeader(const Twine &Text, unsigned int Num, raw_ostream &OS) {
|
2018-10-05 05:34:13 +08:00
|
|
|
OS << std::string(Num, '#') + " " + Text << "\n\n";
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void writeFileDefinition(const ClangDocContext &CDCtx, const Location &L,
|
|
|
|
raw_ostream &OS) {
|
|
|
|
|
|
|
|
if (!CDCtx.RepositoryUrl) {
|
|
|
|
OS << "*Defined at " << L.Filename << "#" << std::to_string(L.LineNumber)
|
|
|
|
<< "*";
|
|
|
|
} else {
|
|
|
|
OS << "*Defined at [" << L.Filename << "#" << std::to_string(L.LineNumber)
|
|
|
|
<< "](" << StringRef{CDCtx.RepositoryUrl.getValue()}
|
|
|
|
<< llvm::sys::path::relative_path(L.Filename) << "#"
|
|
|
|
<< std::to_string(L.LineNumber) << ")"
|
|
|
|
<< "*";
|
|
|
|
}
|
|
|
|
OS << "\n\n";
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
|
2019-07-11 03:03:25 +08:00
|
|
|
static void writeDescription(const CommentInfo &I, raw_ostream &OS) {
|
2018-09-11 23:56:55 +08:00
|
|
|
if (I.Kind == "FullComment") {
|
|
|
|
for (const auto &Child : I.Children)
|
|
|
|
writeDescription(*Child, OS);
|
|
|
|
} else if (I.Kind == "ParagraphComment") {
|
|
|
|
for (const auto &Child : I.Children)
|
|
|
|
writeDescription(*Child, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
} else if (I.Kind == "BlockCommandComment") {
|
|
|
|
OS << genEmphasis(I.Name);
|
|
|
|
for (const auto &Child : I.Children)
|
|
|
|
writeDescription(*Child, OS);
|
|
|
|
} else if (I.Kind == "InlineCommandComment") {
|
|
|
|
OS << genEmphasis(I.Name) << " " << I.Text;
|
|
|
|
} else if (I.Kind == "ParamCommandComment") {
|
|
|
|
std::string Direction = I.Explicit ? (" " + I.Direction).str() : "";
|
2018-10-05 05:34:13 +08:00
|
|
|
OS << genEmphasis(I.ParamName) << I.Text << Direction << "\n\n";
|
2018-09-11 23:56:55 +08:00
|
|
|
} else if (I.Kind == "TParamCommandComment") {
|
|
|
|
std::string Direction = I.Explicit ? (" " + I.Direction).str() : "";
|
2018-10-05 05:34:13 +08:00
|
|
|
OS << genEmphasis(I.ParamName) << I.Text << Direction << "\n\n";
|
2018-09-11 23:56:55 +08:00
|
|
|
} else if (I.Kind == "VerbatimBlockComment") {
|
|
|
|
for (const auto &Child : I.Children)
|
|
|
|
writeDescription(*Child, OS);
|
|
|
|
} else if (I.Kind == "VerbatimBlockLineComment") {
|
|
|
|
OS << I.Text;
|
|
|
|
writeNewLine(OS);
|
|
|
|
} else if (I.Kind == "VerbatimLineComment") {
|
|
|
|
OS << I.Text;
|
|
|
|
writeNewLine(OS);
|
|
|
|
} else if (I.Kind == "HTMLStartTagComment") {
|
|
|
|
if (I.AttrKeys.size() != I.AttrValues.size())
|
|
|
|
return;
|
|
|
|
std::string Buffer;
|
|
|
|
llvm::raw_string_ostream Attrs(Buffer);
|
|
|
|
for (unsigned Idx = 0; Idx < I.AttrKeys.size(); ++Idx)
|
|
|
|
Attrs << " \"" << I.AttrKeys[Idx] << "=" << I.AttrValues[Idx] << "\"";
|
|
|
|
|
|
|
|
std::string CloseTag = I.SelfClosing ? "/>" : ">";
|
|
|
|
writeLine("<" + I.Name + Attrs.str() + CloseTag, OS);
|
|
|
|
} else if (I.Kind == "HTMLEndTagComment") {
|
|
|
|
writeLine("</" + I.Name + ">", OS);
|
|
|
|
} else if (I.Kind == "TextComment") {
|
|
|
|
OS << I.Text;
|
|
|
|
} else {
|
2018-10-05 05:34:13 +08:00
|
|
|
OS << "Unknown comment kind: " << I.Kind << ".\n\n";
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void writeNameLink(const StringRef &CurrentPath, const Reference &R,
|
|
|
|
llvm::raw_ostream &OS) {
|
|
|
|
llvm::SmallString<64> Path = R.getRelativeFilePath(CurrentPath);
|
|
|
|
// Paths in Markdown use POSIX separators.
|
|
|
|
llvm::sys::path::native(Path, llvm::sys::path::Style::posix);
|
|
|
|
llvm::sys::path::append(Path, llvm::sys::path::Style::posix,
|
|
|
|
R.getFileBaseName() + ".md");
|
|
|
|
OS << "[" << R.Name << "](" << Path << ")";
|
|
|
|
}
|
|
|
|
|
|
|
|
static void genMarkdown(const ClangDocContext &CDCtx, const EnumInfo &I,
|
|
|
|
llvm::raw_ostream &OS) {
|
2018-09-11 23:56:55 +08:00
|
|
|
if (I.Scoped)
|
|
|
|
writeLine("| enum class " + I.Name + " |", OS);
|
|
|
|
else
|
|
|
|
writeLine("| enum " + I.Name + " |", OS);
|
|
|
|
writeLine("--", OS);
|
|
|
|
|
|
|
|
std::string Buffer;
|
|
|
|
llvm::raw_string_ostream Members(Buffer);
|
|
|
|
if (!I.Members.empty())
|
|
|
|
for (const auto &N : I.Members)
|
|
|
|
Members << "| " << N << " |\n";
|
|
|
|
writeLine(Members.str(), OS);
|
|
|
|
if (I.DefLoc)
|
2020-03-07 09:33:56 +08:00
|
|
|
writeFileDefinition(CDCtx, I.DefLoc.getValue(), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
|
|
|
|
for (const auto &C : I.Description)
|
|
|
|
writeDescription(C, OS);
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void genMarkdown(const ClangDocContext &CDCtx, const FunctionInfo &I,
|
|
|
|
llvm::raw_ostream &OS) {
|
2018-09-11 23:56:55 +08:00
|
|
|
std::string Buffer;
|
|
|
|
llvm::raw_string_ostream Stream(Buffer);
|
|
|
|
bool First = true;
|
|
|
|
for (const auto &N : I.Params) {
|
|
|
|
if (!First)
|
|
|
|
Stream << ", ";
|
|
|
|
Stream << N.Type.Name + " " + N.Name;
|
|
|
|
First = false;
|
|
|
|
}
|
2018-10-05 05:34:13 +08:00
|
|
|
writeHeader(I.Name, 3, OS);
|
2020-05-28 00:17:07 +08:00
|
|
|
std::string Access = getAccessSpelling(I.Access).str();
|
2018-09-11 23:56:55 +08:00
|
|
|
if (Access != "")
|
2018-10-05 05:34:13 +08:00
|
|
|
writeLine(genItalic(Access + " " + I.ReturnType.Type.Name + " " + I.Name +
|
|
|
|
"(" + Stream.str() + ")"),
|
|
|
|
OS);
|
|
|
|
else
|
|
|
|
writeLine(genItalic(I.ReturnType.Type.Name + " " + I.Name + "(" +
|
|
|
|
Stream.str() + ")"),
|
|
|
|
OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
if (I.DefLoc)
|
2020-03-07 09:33:56 +08:00
|
|
|
writeFileDefinition(CDCtx, I.DefLoc.getValue(), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
|
|
|
|
for (const auto &C : I.Description)
|
|
|
|
writeDescription(C, OS);
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void genMarkdown(const ClangDocContext &CDCtx, const NamespaceInfo &I,
|
|
|
|
llvm::raw_ostream &OS) {
|
2018-09-11 23:56:55 +08:00
|
|
|
if (I.Name == "")
|
|
|
|
writeHeader("Global Namespace", 1, OS);
|
|
|
|
else
|
|
|
|
writeHeader("namespace " + I.Name, 1, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
|
|
|
|
if (!I.Description.empty()) {
|
|
|
|
for (const auto &C : I.Description)
|
|
|
|
writeDescription(C, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
llvm::SmallString<64> BasePath = I.getRelativeFilePath("");
|
|
|
|
|
2018-09-11 23:56:55 +08:00
|
|
|
if (!I.ChildNamespaces.empty()) {
|
|
|
|
writeHeader("Namespaces", 2, OS);
|
2020-03-07 09:33:56 +08:00
|
|
|
for (const auto &R : I.ChildNamespaces) {
|
|
|
|
OS << "* ";
|
|
|
|
writeNameLink(BasePath, R, OS);
|
|
|
|
OS << "\n";
|
|
|
|
}
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
2020-03-07 09:33:56 +08:00
|
|
|
|
2018-09-11 23:56:55 +08:00
|
|
|
if (!I.ChildRecords.empty()) {
|
|
|
|
writeHeader("Records", 2, OS);
|
2020-03-07 09:33:56 +08:00
|
|
|
for (const auto &R : I.ChildRecords) {
|
|
|
|
OS << "* ";
|
|
|
|
writeNameLink(BasePath, R, OS);
|
|
|
|
OS << "\n";
|
|
|
|
}
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
2020-03-07 09:33:56 +08:00
|
|
|
|
2018-09-11 23:56:55 +08:00
|
|
|
if (!I.ChildFunctions.empty()) {
|
|
|
|
writeHeader("Functions", 2, OS);
|
|
|
|
for (const auto &F : I.ChildFunctions)
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, F, OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
if (!I.ChildEnums.empty()) {
|
|
|
|
writeHeader("Enums", 2, OS);
|
|
|
|
for (const auto &E : I.ChildEnums)
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, E, OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void genMarkdown(const ClangDocContext &CDCtx, const RecordInfo &I,
|
|
|
|
llvm::raw_ostream &OS) {
|
2018-09-11 23:56:55 +08:00
|
|
|
writeHeader(getTagType(I.TagType) + " " + I.Name, 1, OS);
|
|
|
|
if (I.DefLoc)
|
2020-03-07 09:33:56 +08:00
|
|
|
writeFileDefinition(CDCtx, I.DefLoc.getValue(), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
|
|
|
|
if (!I.Description.empty()) {
|
|
|
|
for (const auto &C : I.Description)
|
|
|
|
writeDescription(C, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Parents = genReferenceList(I.Parents);
|
|
|
|
std::string VParents = genReferenceList(I.VirtualParents);
|
|
|
|
if (!Parents.empty() || !VParents.empty()) {
|
|
|
|
if (Parents.empty())
|
|
|
|
writeLine("Inherits from " + VParents, OS);
|
|
|
|
else if (VParents.empty())
|
|
|
|
writeLine("Inherits from " + Parents, OS);
|
|
|
|
else
|
|
|
|
writeLine("Inherits from " + Parents + ", " + VParents, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!I.Members.empty()) {
|
|
|
|
writeHeader("Members", 2, OS);
|
2020-01-02 00:23:21 +08:00
|
|
|
for (const auto &Member : I.Members) {
|
2020-05-28 00:17:07 +08:00
|
|
|
std::string Access = getAccessSpelling(Member.Access).str();
|
2018-09-11 23:56:55 +08:00
|
|
|
if (Access != "")
|
|
|
|
writeLine(Access + " " + Member.Type.Name + " " + Member.Name, OS);
|
|
|
|
else
|
|
|
|
writeLine(Member.Type.Name + " " + Member.Name, OS);
|
|
|
|
}
|
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!I.ChildRecords.empty()) {
|
|
|
|
writeHeader("Records", 2, OS);
|
|
|
|
for (const auto &R : I.ChildRecords)
|
|
|
|
writeLine(R.Name, OS);
|
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
if (!I.ChildFunctions.empty()) {
|
|
|
|
writeHeader("Functions", 2, OS);
|
|
|
|
for (const auto &F : I.ChildFunctions)
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, F, OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
if (!I.ChildEnums.empty()) {
|
|
|
|
writeHeader("Enums", 2, OS);
|
|
|
|
for (const auto &E : I.ChildEnums)
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, E, OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
writeNewLine(OS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
static void serializeReference(llvm::raw_fd_ostream &OS, Index &I, int Level) {
|
|
|
|
// Write out the heading level starting at ##
|
|
|
|
OS << "##" << std::string(Level, '#') << " ";
|
|
|
|
writeNameLink("", I, OS);
|
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
static llvm::Error serializeIndex(ClangDocContext &CDCtx) {
|
|
|
|
std::error_code FileErr;
|
|
|
|
llvm::SmallString<128> FilePath;
|
|
|
|
llvm::sys::path::native(CDCtx.OutDirectory, FilePath);
|
|
|
|
llvm::sys::path::append(FilePath, "all_files.md");
|
|
|
|
llvm::raw_fd_ostream OS(FilePath, FileErr, llvm::sys::fs::OF_None);
|
|
|
|
if (FileErr)
|
|
|
|
return llvm::createStringError(llvm::inconvertibleErrorCode(),
|
|
|
|
"error creating index file: " +
|
|
|
|
FileErr.message());
|
|
|
|
|
|
|
|
CDCtx.Idx.sort();
|
|
|
|
OS << "# All Files";
|
|
|
|
if (!CDCtx.ProjectName.empty())
|
|
|
|
OS << " for " << CDCtx.ProjectName;
|
|
|
|
OS << "\n\n";
|
|
|
|
|
|
|
|
for (auto C : CDCtx.Idx.Children)
|
|
|
|
serializeReference(OS, C, 0);
|
|
|
|
|
|
|
|
return llvm::Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
static llvm::Error genIndex(ClangDocContext &CDCtx) {
|
|
|
|
std::error_code FileErr;
|
|
|
|
llvm::SmallString<128> FilePath;
|
|
|
|
llvm::sys::path::native(CDCtx.OutDirectory, FilePath);
|
|
|
|
llvm::sys::path::append(FilePath, "index.md");
|
|
|
|
llvm::raw_fd_ostream OS(FilePath, FileErr, llvm::sys::fs::OF_None);
|
|
|
|
if (FileErr)
|
|
|
|
return llvm::createStringError(llvm::inconvertibleErrorCode(),
|
|
|
|
"error creating index file: " +
|
|
|
|
FileErr.message());
|
|
|
|
CDCtx.Idx.sort();
|
|
|
|
OS << "# " << CDCtx.ProjectName << " C/C++ Reference\n\n";
|
|
|
|
for (auto C : CDCtx.Idx.Children) {
|
|
|
|
if (!C.Children.empty()) {
|
|
|
|
const char *Type;
|
|
|
|
switch (C.RefType) {
|
|
|
|
case InfoType::IT_namespace:
|
|
|
|
Type = "Namespace";
|
|
|
|
break;
|
|
|
|
case InfoType::IT_record:
|
|
|
|
Type = "Type";
|
|
|
|
break;
|
|
|
|
case InfoType::IT_enum:
|
|
|
|
Type = "Enum";
|
|
|
|
break;
|
|
|
|
case InfoType::IT_function:
|
|
|
|
Type = "Function";
|
|
|
|
break;
|
|
|
|
case InfoType::IT_default:
|
|
|
|
Type = "Other";
|
|
|
|
}
|
|
|
|
OS << "* " << Type << ": [" << C.Name << "](";
|
|
|
|
if (!C.Path.empty())
|
|
|
|
OS << C.Path << "/";
|
|
|
|
OS << C.Name << ")\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return llvm::Error::success();
|
|
|
|
}
|
2018-09-11 23:56:55 +08:00
|
|
|
/// Generator for Markdown documentation.
|
|
|
|
class MDGenerator : public Generator {
|
|
|
|
public:
|
|
|
|
static const char *Format;
|
|
|
|
|
2019-07-26 06:46:40 +08:00
|
|
|
llvm::Error generateDocForInfo(Info *I, llvm::raw_ostream &OS,
|
|
|
|
const ClangDocContext &CDCtx) override;
|
2020-03-07 09:33:56 +08:00
|
|
|
llvm::Error createResources(ClangDocContext &CDCtx) override;
|
2018-09-11 23:56:55 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
const char *MDGenerator::Format = "md";
|
|
|
|
|
2019-07-26 06:46:40 +08:00
|
|
|
llvm::Error MDGenerator::generateDocForInfo(Info *I, llvm::raw_ostream &OS,
|
|
|
|
const ClangDocContext &CDCtx) {
|
2018-09-11 23:56:55 +08:00
|
|
|
switch (I->IT) {
|
|
|
|
case InfoType::IT_namespace:
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, *static_cast<clang::doc::NamespaceInfo *>(I), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
break;
|
|
|
|
case InfoType::IT_record:
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, *static_cast<clang::doc::RecordInfo *>(I), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
break;
|
|
|
|
case InfoType::IT_enum:
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, *static_cast<clang::doc::EnumInfo *>(I), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
break;
|
|
|
|
case InfoType::IT_function:
|
2020-03-07 09:33:56 +08:00
|
|
|
genMarkdown(CDCtx, *static_cast<clang::doc::FunctionInfo *>(I), OS);
|
2018-09-11 23:56:55 +08:00
|
|
|
break;
|
|
|
|
case InfoType::IT_default:
|
2019-08-28 10:56:03 +08:00
|
|
|
return createStringError(llvm::inconvertibleErrorCode(),
|
|
|
|
"unexpected InfoType");
|
2018-09-11 23:56:55 +08:00
|
|
|
}
|
|
|
|
return llvm::Error::success();
|
|
|
|
}
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
llvm::Error MDGenerator::createResources(ClangDocContext &CDCtx) {
|
|
|
|
// Write an all_files.md
|
|
|
|
auto Err = serializeIndex(CDCtx);
|
|
|
|
if (Err)
|
|
|
|
return Err;
|
|
|
|
|
|
|
|
// Generate the index page.
|
|
|
|
Err = genIndex(CDCtx);
|
|
|
|
if (Err)
|
|
|
|
return Err;
|
|
|
|
|
|
|
|
return llvm::Error::success();
|
|
|
|
}
|
|
|
|
|
2018-09-11 23:56:55 +08:00
|
|
|
static GeneratorRegistry::Add<MDGenerator> MD(MDGenerator::Format,
|
|
|
|
"Generator for MD output.");
|
|
|
|
|
2020-03-07 09:33:56 +08:00
|
|
|
// This anchor is used to force the linker to link in the generated object
|
|
|
|
// file and thus register the generator.
|
2018-09-11 23:56:55 +08:00
|
|
|
volatile int MDGeneratorAnchorSource = 0;
|
|
|
|
|
|
|
|
} // namespace doc
|
|
|
|
} // namespace clang
|