clang-tools 20.0.0git
ClangDocMain.cpp
Go to the documentation of this file.
1//===-- ClangDocMain.cpp - ClangDoc -----------------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This tool for generating C and C++ documentation from source code
10// and comments. Generally, it runs a LibTooling FrontendAction on source files,
11// mapping each declaration in those files to its USR and serializing relevant
12// information into LLVM bitcode. It then runs a pass over the collected
13// declaration information, reducing by USR. There is an option to dump this
14// intermediate result to bitcode. Finally, it hands the reduced information
15// off to a generator, which does the final parsing from the intermediate
16// representation to the desired output format.
17//
18//===----------------------------------------------------------------------===//
19
20#include "BitcodeReader.h"
21#include "BitcodeWriter.h"
22#include "ClangDoc.h"
23#include "Generators.h"
24#include "Representation.h"
25#include "clang/AST/AST.h"
26#include "clang/AST/Decl.h"
27#include "clang/ASTMatchers/ASTMatchFinder.h"
28#include "clang/ASTMatchers/ASTMatchersInternal.h"
29#include "clang/Driver/Options.h"
30#include "clang/Frontend/FrontendActions.h"
31#include "clang/Tooling/AllTUsExecution.h"
32#include "clang/Tooling/CommonOptionsParser.h"
33#include "clang/Tooling/Execution.h"
34#include "clang/Tooling/Tooling.h"
35#include "llvm/ADT/APFloat.h"
36#include "llvm/Support/CommandLine.h"
37#include "llvm/Support/Error.h"
38#include "llvm/Support/FileSystem.h"
39#include "llvm/Support/Mutex.h"
40#include "llvm/Support/Path.h"
41#include "llvm/Support/Process.h"
42#include "llvm/Support/Signals.h"
43#include "llvm/Support/ThreadPool.h"
44#include "llvm/Support/raw_ostream.h"
45#include <atomic>
46#include <mutex>
47#include <string>
48
49using namespace clang::ast_matchers;
50using namespace clang::tooling;
51using namespace clang;
52
53static llvm::cl::extrahelp CommonHelp(CommonOptionsParser::HelpMessage);
54static llvm::cl::OptionCategory ClangDocCategory("clang-doc options");
55
56static llvm::cl::opt<std::string>
57 ProjectName("project-name", llvm::cl::desc("Name of project."),
58 llvm::cl::cat(ClangDocCategory));
59
60static llvm::cl::opt<bool> IgnoreMappingFailures(
61 "ignore-map-errors",
62 llvm::cl::desc("Continue if files are not mapped correctly."),
63 llvm::cl::init(true), llvm::cl::cat(ClangDocCategory));
64
65static llvm::cl::opt<std::string>
66 OutDirectory("output",
67 llvm::cl::desc("Directory for outputting generated files."),
68 llvm::cl::init("docs"), llvm::cl::cat(ClangDocCategory));
69
70static llvm::cl::opt<bool>
71 PublicOnly("public", llvm::cl::desc("Document only public declarations."),
72 llvm::cl::init(false), llvm::cl::cat(ClangDocCategory));
73
74static llvm::cl::opt<bool> DoxygenOnly(
75 "doxygen",
76 llvm::cl::desc("Use only doxygen-style comments to generate docs."),
77 llvm::cl::init(false), llvm::cl::cat(ClangDocCategory));
78
79static llvm::cl::list<std::string> UserStylesheets(
80 "stylesheets", llvm::cl::CommaSeparated,
81 llvm::cl::desc("CSS stylesheets to extend the default styles."),
82 llvm::cl::cat(ClangDocCategory));
83
84static llvm::cl::opt<std::string> UserAssetPath(
85 "asset",
86 llvm::cl::desc("User supplied asset path to "
87 "override the default css and js files for html output"),
88 llvm::cl::cat(ClangDocCategory));
89
90static llvm::cl::opt<std::string> SourceRoot("source-root", llvm::cl::desc(R"(
91Directory where processed files are stored.
92Links to definition locations will only be
93generated if the file is in this dir.)"),
94 llvm::cl::cat(ClangDocCategory));
95
96static llvm::cl::opt<std::string>
97 RepositoryUrl("repository", llvm::cl::desc(R"(
98URL of repository that hosts code.
99Used for links to definition locations.)"),
100 llvm::cl::cat(ClangDocCategory));
101
102enum OutputFormatTy {
103 md,
105 html,
106};
107
108static llvm::cl::opt<OutputFormatTy>
109 FormatEnum("format", llvm::cl::desc("Format for outputted docs."),
110 llvm::cl::values(clEnumValN(OutputFormatTy::yaml, "yaml",
111 "Documentation in YAML format."),
112 clEnumValN(OutputFormatTy::md, "md",
113 "Documentation in MD format."),
114 clEnumValN(OutputFormatTy::html, "html",
115 "Documentation in HTML format.")),
116 llvm::cl::init(OutputFormatTy::yaml),
117 llvm::cl::cat(ClangDocCategory));
118
119std::string getFormatString() {
120 switch (FormatEnum) {
122 return "yaml";
124 return "md";
126 return "html";
127 }
128 llvm_unreachable("Unknown OutputFormatTy");
129}
130
131// This function isn't referenced outside its translation unit, but it
132// can't use the "static" keyword because its address is used for
133// GetMainExecutable (since some platforms don't support taking the
134// address of main, and some platforms can't implement GetMainExecutable
135// without being given the address of a function in the main executable).
136std::string getExecutablePath(const char *Argv0, void *MainAddr) {
137 return llvm::sys::fs::getMainExecutable(Argv0, MainAddr);
138}
139
140llvm::Error getAssetFiles(clang::doc::ClangDocContext &CDCtx) {
141 using DirIt = llvm::sys::fs::directory_iterator;
142 std::error_code FileErr;
143 llvm::SmallString<128> FilePath(UserAssetPath);
144 for (DirIt DirStart = DirIt(UserAssetPath, FileErr),
145 DirEnd;
146 !FileErr && DirStart != DirEnd; DirStart.increment(FileErr)) {
147 FilePath = DirStart->path();
148 if (llvm::sys::fs::is_regular_file(FilePath)) {
149 if (llvm::sys::path::extension(FilePath) == ".css")
150 CDCtx.UserStylesheets.insert(CDCtx.UserStylesheets.begin(),
151 std::string(FilePath));
152 else if (llvm::sys::path::extension(FilePath) == ".js")
153 CDCtx.JsScripts.emplace_back(FilePath.str());
154 }
155 }
156 if (FileErr)
157 return llvm::createFileError(FilePath, FileErr);
158 return llvm::Error::success();
159}
160
161llvm::Error getDefaultAssetFiles(const char *Argv0,
163 void *MainAddr = (void *)(intptr_t)getExecutablePath;
164 std::string ClangDocPath = getExecutablePath(Argv0, MainAddr);
165 llvm::SmallString<128> NativeClangDocPath;
166 llvm::sys::path::native(ClangDocPath, NativeClangDocPath);
167
168 llvm::SmallString<128> AssetsPath;
169 AssetsPath = llvm::sys::path::parent_path(NativeClangDocPath);
170 llvm::sys::path::append(AssetsPath, "..", "share", "clang-doc");
171 llvm::SmallString<128> DefaultStylesheet;
172 llvm::sys::path::native(AssetsPath, DefaultStylesheet);
173 llvm::sys::path::append(DefaultStylesheet,
174 "clang-doc-default-stylesheet.css");
175 llvm::SmallString<128> IndexJS;
176 llvm::sys::path::native(AssetsPath, IndexJS);
177 llvm::sys::path::append(IndexJS, "index.js");
178
179 if (!llvm::sys::fs::is_regular_file(IndexJS))
180 return llvm::createStringError(llvm::inconvertibleErrorCode(),
181 "default index.js file missing at " +
182 IndexJS + "\n");
183
184 if (!llvm::sys::fs::is_regular_file(DefaultStylesheet))
185 return llvm::createStringError(
186 llvm::inconvertibleErrorCode(),
187 "default clang-doc-default-stylesheet.css file missing at " +
188 DefaultStylesheet + "\n");
189
190 CDCtx.UserStylesheets.insert(CDCtx.UserStylesheets.begin(),
191 std::string(DefaultStylesheet));
192 CDCtx.JsScripts.emplace_back(IndexJS.str());
193
194 return llvm::Error::success();
195}
196
197llvm::Error getHtmlAssetFiles(const char *Argv0,
199 if (!UserAssetPath.empty() &&
200 !llvm::sys::fs::is_directory(std::string(UserAssetPath)))
201 llvm::outs() << "Asset path supply is not a directory: " << UserAssetPath
202 << " falling back to default\n";
203 if (llvm::sys::fs::is_directory(std::string(UserAssetPath)))
204 return getAssetFiles(CDCtx);
205 return getDefaultAssetFiles(Argv0, CDCtx);
206}
207
208int main(int argc, const char **argv) {
209 llvm::sys::PrintStackTraceOnErrorSignal(argv[0]);
210 std::error_code OK;
211
212 const char *Overview =
213 R"(Generates documentation from source code and comments.
214
215Example usage for files without flags (default):
216
217 $ clang-doc File1.cpp File2.cpp ... FileN.cpp
218
219Example usage for a project using a compile commands database:
220
221 $ clang-doc --executor=all-TUs compile_commands.json
222)";
223
224 auto Executor = clang::tooling::createExecutorFromCommandLineArgs(
225 argc, argv, ClangDocCategory, Overview);
226
227 if (!Executor) {
228 llvm::errs() << toString(Executor.takeError()) << "\n";
229 return 1;
230 }
231
232 // Fail early if an invalid format was provided.
233 std::string Format = getFormatString();
234 llvm::outs() << "Emiting docs in " << Format << " format.\n";
235 auto G = doc::findGeneratorByName(Format);
236 if (!G) {
237 llvm::errs() << toString(G.takeError()) << "\n";
238 return 1;
239 }
240
241 ArgumentsAdjuster ArgAdjuster;
242 if (!DoxygenOnly)
243 ArgAdjuster = combineAdjusters(
244 getInsertArgumentAdjuster("-fparse-all-comments",
245 tooling::ArgumentInsertPosition::END),
246 ArgAdjuster);
247
249 Executor->get()->getExecutionContext(),
255 {UserStylesheets.begin(), UserStylesheets.end()}
256 };
257
258 if (Format == "html") {
259 if (auto Err = getHtmlAssetFiles(argv[0], CDCtx)) {
260 llvm::errs() << toString(std::move(Err)) << "\n";
261 return 1;
262 }
263 }
264
265 // Mapping phase
266 llvm::outs() << "Mapping decls...\n";
267 auto Err =
268 Executor->get()->execute(doc::newMapperActionFactory(CDCtx), ArgAdjuster);
269 if (Err) {
271 llvm::errs() << "Error mapping decls in files. Clang-doc will ignore "
272 "these files and continue:\n"
273 << toString(std::move(Err)) << "\n";
274 else {
275 llvm::errs() << toString(std::move(Err)) << "\n";
276 return 1;
277 }
278 }
279
280 // Collect values into output by key.
281 // In ToolResults, the Key is the hashed USR and the value is the
282 // bitcode-encoded representation of the Info object.
283 llvm::outs() << "Collecting infos...\n";
284 llvm::StringMap<std::vector<StringRef>> USRToBitcode;
285 Executor->get()->getToolResults()->forEachResult(
286 [&](StringRef Key, StringRef Value) {
287 auto R = USRToBitcode.try_emplace(Key, std::vector<StringRef>());
288 R.first->second.emplace_back(Value);
289 });
290
291 // Collects all Infos according to their unique USR value. This map is added
292 // to from the thread pool below and is protected by the USRToInfoMutex.
293 llvm::sys::Mutex USRToInfoMutex;
294 llvm::StringMap<std::unique_ptr<doc::Info>> USRToInfo;
295
296 // First reducing phase (reduce all decls into one info per decl).
297 llvm::outs() << "Reducing " << USRToBitcode.size() << " infos...\n";
298 std::atomic<bool> Error;
299 Error = false;
300 llvm::sys::Mutex IndexMutex;
301 // ExecutorConcurrency is a flag exposed by AllTUsExecution.h
302 llvm::DefaultThreadPool Pool(llvm::hardware_concurrency(ExecutorConcurrency));
303 for (auto &Group : USRToBitcode) {
304 Pool.async([&]() {
305 std::vector<std::unique_ptr<doc::Info>> Infos;
306 for (auto &Bitcode : Group.getValue()) {
307 llvm::BitstreamCursor Stream(Bitcode);
308 doc::ClangDocBitcodeReader Reader(Stream);
309 auto ReadInfos = Reader.readBitcode();
310 if (!ReadInfos) {
311 llvm::errs() << toString(ReadInfos.takeError()) << "\n";
312 Error = true;
313 return;
314 }
315 std::move(ReadInfos->begin(), ReadInfos->end(),
316 std::back_inserter(Infos));
317 }
318
319 auto Reduced = doc::mergeInfos(Infos);
320 if (!Reduced) {
321 llvm::errs() << llvm::toString(Reduced.takeError());
322 return;
323 }
324
325 // Add a reference to this Info in the Index
326 {
327 std::lock_guard<llvm::sys::Mutex> Guard(IndexMutex);
328 clang::doc::Generator::addInfoToIndex(CDCtx.Idx, Reduced.get().get());
329 }
330
331 // Save in the result map (needs a lock due to threaded access).
332 {
333 std::lock_guard<llvm::sys::Mutex> Guard(USRToInfoMutex);
334 USRToInfo[Group.getKey()] = std::move(Reduced.get());
335 }
336 });
337 }
338
339 Pool.wait();
340
341 if (Error)
342 return 1;
343
344 // Ensure the root output directory exists.
345 if (std::error_code Err = llvm::sys::fs::create_directories(OutDirectory);
346 Err != std::error_code()) {
347 llvm::errs() << "Failed to create directory '" << OutDirectory << "'\n";
348 return 1;
349 }
350
351 // Run the generator.
352 llvm::outs() << "Generating docs...\n";
353 if (auto Err =
354 G->get()->generateDocs(OutDirectory, std::move(USRToInfo), CDCtx)) {
355 llvm::errs() << toString(std::move(Err)) << "\n";
356 return 1;
357 }
358
359 llvm::outs() << "Generating assets for docs...\n";
360 Err = G->get()->createResources(CDCtx);
361 if (Err) {
362 llvm::outs() << "warning: " << toString(std::move(Err)) << "\n";
363 }
364
365 return 0;
366}
static llvm::cl::opt< std::string > UserAssetPath("asset", llvm::cl::desc("User supplied asset path to " "override the default css and js files for html output"), llvm::cl::cat(ClangDocCategory))
static llvm::cl::extrahelp CommonHelp(CommonOptionsParser::HelpMessage)
static llvm::cl::opt< bool > PublicOnly("public", llvm::cl::desc("Document only public declarations."), llvm::cl::init(false), llvm::cl::cat(ClangDocCategory))
std::string getExecutablePath(const char *Argv0, void *MainAddr)
int main(int argc, const char **argv)
llvm::Error getHtmlAssetFiles(const char *Argv0, clang::doc::ClangDocContext &CDCtx)
static llvm::cl::opt< std::string > ProjectName("project-name", llvm::cl::desc("Name of project."), llvm::cl::cat(ClangDocCategory))
static llvm::cl::opt< OutputFormatTy > FormatEnum("format", llvm::cl::desc("Format for outputted docs."), llvm::cl::values(clEnumValN(OutputFormatTy::yaml, "yaml", "Documentation in YAML format."), clEnumValN(OutputFormatTy::md, "md", "Documentation in MD format."), clEnumValN(OutputFormatTy::html, "html", "Documentation in HTML format.")), llvm::cl::init(OutputFormatTy::yaml), llvm::cl::cat(ClangDocCategory))
OutputFormatTy
@ yaml
@ html
@ md
static llvm::cl::list< std::string > UserStylesheets("stylesheets", llvm::cl::CommaSeparated, llvm::cl::desc("CSS stylesheets to extend the default styles."), llvm::cl::cat(ClangDocCategory))
llvm::Error getAssetFiles(clang::doc::ClangDocContext &CDCtx)
llvm::Error getDefaultAssetFiles(const char *Argv0, clang::doc::ClangDocContext &CDCtx)
static llvm::cl::opt< std::string > SourceRoot("source-root", llvm::cl::desc(R"( Directory where processed files are stored. Links to definition locations will only be generated if the file is in this dir.)"), llvm::cl::cat(ClangDocCategory))
static llvm::cl::opt< bool > IgnoreMappingFailures("ignore-map-errors", llvm::cl::desc("Continue if files are not mapped correctly."), llvm::cl::init(true), llvm::cl::cat(ClangDocCategory))
static llvm::cl::opt< std::string > OutDirectory("output", llvm::cl::desc("Directory for outputting generated files."), llvm::cl::init("docs"), llvm::cl::cat(ClangDocCategory))
static llvm::cl::opt< bool > DoxygenOnly("doxygen", llvm::cl::desc("Use only doxygen-style comments to generate docs."), llvm::cl::init(false), llvm::cl::cat(ClangDocCategory))
static llvm::cl::OptionCategory ClangDocCategory("clang-doc options")
static llvm::cl::opt< std::string > RepositoryUrl("repository", llvm::cl::desc(R"( URL of repository that hosts code. Used for links to definition locations.)"), llvm::cl::cat(ClangDocCategory))
std::string getFormatString()
static constexpr llvm::SourceMgr::DiagKind Error
const char * Argv0
Definition: Modularize.cpp:333
static void addInfoToIndex(Index &Idx, const doc::Info *Info)
Definition: Generators.cpp:59
llvm::Expected< std::unique_ptr< Info > > mergeInfos(std::vector< std::unique_ptr< Info > > &Values)
std::unique_ptr< tooling::FrontendActionFactory > newMapperActionFactory(ClangDocContext CDCtx)
Definition: ClangDoc.cpp:56
llvm::Expected< std::unique_ptr< Generator > > findGeneratorByName(llvm::StringRef Format)
Definition: Generators.cpp:17
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
std::vector< std::string > UserStylesheets
std::vector< std::string > JsScripts