Files
llvm-project/clang-tools-extra/clang-doc/HTMLGenerator.cpp
Paul Kirth 744d8eff65 [clang-doc] Move non-arena allocated types off the OwnedPtr alias (#190052)
Some types should not be using this alias, which was over applied to
APIs that wont participate in arena style allocation. This patch
restores them to their correct spelling.
2026-04-10 20:38:55 +00:00

203 lines
7.9 KiB
C++

//===-- HTMLGenerator.cpp - HTML Generator ----------------------*- 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
//
//===----------------------------------------------------------------------===//
///
/// \file
/// This file contains the implementation of the HTMLGenerator class,
/// which is a Clang-Doc generator for HTML using Mustache templates.
///
//===----------------------------------------------------------------------===//
#include "Generators.h"
#include "Representation.h"
#include "support/File.h"
#include "llvm/Support/Error.h"
#include "llvm/Support/Path.h"
using namespace llvm;
using namespace llvm::json;
using namespace llvm::mustache;
namespace clang {
namespace doc {
static std::unique_ptr<MustacheTemplateFile> NamespaceTemplate = nullptr;
static std::unique_ptr<MustacheTemplateFile> RecordTemplate = nullptr;
static std::unique_ptr<MustacheTemplateFile> IndexTemplate = nullptr;
class HTMLGenerator : public MustacheGenerator {
public:
static const char *Format;
Error createResources(ClangDocContext &CDCtx) override;
Error generateDocForInfo(Info *I, raw_ostream &OS,
const ClangDocContext &CDCtx) override;
Error setupTemplateFiles(const ClangDocContext &CDCtx) override;
Error generateDocForJSON(json::Value &JSON, raw_fd_ostream &OS,
const ClangDocContext &CDCtx, StringRef ObjTypeStr,
StringRef RelativeRootPath) override;
// Populates templates with CSS stylesheets, JS scripts paths.
Error setupTemplateResources(const ClangDocContext &CDCtx, json::Value &V,
SmallString<128> RelativeRootPath);
llvm::Error generateDocumentation(
StringRef RootDir, llvm::StringMap<doc::OwnedPtr<doc::Info>> Infos,
const ClangDocContext &CDCtx, std::string DirName) override;
};
Error HTMLGenerator::setupTemplateFiles(const ClangDocContext &CDCtx) {
// Template files need to use the native path when they're opened,
// but have to be used in POSIX style when used in HTML.
auto ConvertToNative = [](std::string &&Path) -> std::string {
SmallString<128> PathBuf(Path);
llvm::sys::path::native(PathBuf);
return PathBuf.str().str();
};
std::string NamespaceFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("namespace-template"));
std::string ClassFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("class-template"));
std::string IndexFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("index-template"));
std::string CommentFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("comment-template"));
std::string FunctionFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("function-template"));
std::string EnumFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("enum-template"));
std::string HeadFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("head-template"));
std::string NavbarFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("navbar-template"));
std::string AliasFilePath =
ConvertToNative(CDCtx.MustacheTemplates.lookup("alias-template"));
std::vector<std::pair<StringRef, StringRef>> Partials = {
{"Comments", CommentFilePath}, {"FunctionPartial", FunctionFilePath},
{"EnumPartial", EnumFilePath}, {"HeadPartial", HeadFilePath},
{"NavbarPartial", NavbarFilePath}, {"AliasPartial", AliasFilePath}};
if (Error Err = setupTemplate(NamespaceTemplate, NamespaceFilePath, Partials))
return Err;
if (Error Err = setupTemplate(RecordTemplate, ClassFilePath, Partials))
return Err;
if (Error Err = setupTemplate(IndexTemplate, IndexFilePath, Partials))
return Err;
return Error::success();
}
Error HTMLGenerator::setupTemplateResources(const ClangDocContext &CDCtx,
json::Value &V,
SmallString<128> RelativeRootPath) {
V.getAsObject()->insert({"ProjectName", CDCtx.ProjectName});
json::Value StylesheetArr = Array();
sys::path::native(RelativeRootPath, sys::path::Style::posix);
auto *SSA = StylesheetArr.getAsArray();
SSA->reserve(CDCtx.UserStylesheets.size());
for (const auto &FilePath : CDCtx.UserStylesheets) {
SmallString<128> StylesheetPath = RelativeRootPath;
sys::path::append(StylesheetPath, sys::path::Style::posix,
sys::path::filename(FilePath));
SSA->emplace_back(StylesheetPath);
}
V.getAsObject()->insert({"Stylesheets", StylesheetArr});
json::Value ScriptArr = Array();
auto *SCA = ScriptArr.getAsArray();
SCA->reserve(CDCtx.JsScripts.size());
for (auto Script : CDCtx.JsScripts) {
SmallString<128> JsPath = RelativeRootPath;
sys::path::append(JsPath, sys::path::Style::posix,
sys::path::filename(Script));
SCA->emplace_back(JsPath);
}
V.getAsObject()->insert({"Scripts", ScriptArr});
if (RelativeRootPath.empty()) {
RelativeRootPath = "";
} else {
sys::path::append(RelativeRootPath, "/index.html");
sys::path::native(RelativeRootPath, sys::path::Style::posix);
}
V.getAsObject()->insert({"Homepage", RelativeRootPath});
return Error::success();
}
Error HTMLGenerator::generateDocForJSON(json::Value &JSON, raw_fd_ostream &OS,
const ClangDocContext &CDCtx,
StringRef ObjTypeStr,
StringRef RelativeRootPath) {
if (ObjTypeStr == "namespace") {
if (auto Err = setupTemplateResources(CDCtx, JSON, RelativeRootPath))
return Err;
assert(NamespaceTemplate && "NamespaceTemplate is nullptr.");
NamespaceTemplate->render(JSON, OS);
} else if (ObjTypeStr == "record") {
if (auto Err = setupTemplateResources(CDCtx, JSON, RelativeRootPath))
return Err;
assert(RecordTemplate && "RecordTemplate is nullptr.");
RecordTemplate->render(JSON, OS);
} else if (ObjTypeStr == "index") {
if (auto Err = setupTemplateResources(CDCtx, JSON, RelativeRootPath))
return Err;
assert(IndexTemplate && "IndexTemplate is nullptr.");
IndexTemplate->render(JSON, OS);
}
return Error::success();
}
Error HTMLGenerator::generateDocForInfo(Info *I, raw_ostream &OS,
const ClangDocContext &CDCtx) {
switch (I->IT) {
case InfoType::IT_enum:
case InfoType::IT_function:
case InfoType::IT_typedef:
case InfoType::IT_namespace:
case InfoType::IT_record:
case InfoType::IT_concept:
case InfoType::IT_variable:
case InfoType::IT_friend:
break;
case InfoType::IT_default:
return createStringError(inconvertibleErrorCode(), "unexpected InfoType");
}
return Error::success();
}
Error HTMLGenerator::createResources(ClangDocContext &CDCtx) {
std::string ResourcePath(CDCtx.OutDirectory + "/html");
for (const auto &FilePath : CDCtx.UserStylesheets)
if (Error Err = copyFile(FilePath, ResourcePath))
return Err;
for (const auto &FilePath : CDCtx.JsScripts)
if (Error Err = copyFile(FilePath, ResourcePath))
return Err;
return Error::success();
}
Error HTMLGenerator::generateDocumentation(
StringRef RootDir, llvm::StringMap<doc::OwnedPtr<doc::Info>> Infos,
const ClangDocContext &CDCtx, std::string DirName) {
return MustacheGenerator::generateDocumentation(RootDir, std::move(Infos),
CDCtx, "html");
}
const char *HTMLGenerator::Format = "html";
static GeneratorRegistry::Add<HTMLGenerator>
HTML(HTMLGenerator::Format, "Generator for mustache HTML output.");
// This anchor is used to force the linker to link in the generated object
// file and thus register the generator.
volatile int HTMLGeneratorAnchorSource = 0;
} // namespace doc
} // namespace clang