//===-- ClangDocMain.cpp - ClangDoc -----------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This tool for generating C and C++ documenation from source code // and comments. Generally, it runs a LibTooling FrontendAction on source files, // mapping each declaration in those files to its USR and serializing relevant // information into LLVM bitcode. It then runs a pass over the collected // declaration information, reducing by USR. There is an option to dump this // intermediate result to bitcode. Finally, it hands the reduced information // off to a generator, which does the final parsing from the intermediate // representation to the desired output format. // //===----------------------------------------------------------------------===// #include "BitcodeReader.h" #include "BitcodeWriter.h" #include "ClangDoc.h" #include "Generators.h" #include "Representation.h" #include "clang/AST/AST.h" #include "clang/AST/Decl.h" #include "clang/ASTMatchers/ASTMatchFinder.h" #include "clang/ASTMatchers/ASTMatchersInternal.h" #include "clang/Driver/Options.h" #include "clang/Frontend/FrontendActions.h" #include "clang/Tooling/CommonOptionsParser.h" #include "clang/Tooling/Execution.h" #include "clang/Tooling/StandaloneExecution.h" #include "clang/Tooling/Tooling.h" #include "llvm/ADT/APFloat.h" #include "llvm/Support/Error.h" #include "llvm/Support/FileSystem.h" #include "llvm/Support/Path.h" #include "llvm/Support/Process.h" #include "llvm/Support/Signals.h" #include "llvm/Support/raw_ostream.h" #include using namespace clang::ast_matchers; using namespace clang::tooling; using namespace clang; static llvm::cl::extrahelp CommonHelp(CommonOptionsParser::HelpMessage); static llvm::cl::OptionCategory ClangDocCategory("clang-doc options"); static llvm::cl::opt OutDirectory("output", llvm::cl::desc("Directory for outputting generated files."), llvm::cl::init("docs"), llvm::cl::cat(ClangDocCategory)); static llvm::cl::opt DumpMapperResult("dump-mapper", llvm::cl::desc("Dump mapper results to bitcode file."), llvm::cl::init(false), llvm::cl::cat(ClangDocCategory)); static llvm::cl::opt DumpIntermediateResult( "dump-intermediate", llvm::cl::desc("Dump intermediate results to bitcode file."), llvm::cl::init(false), llvm::cl::cat(ClangDocCategory)); enum OutputFormatTy { yaml, }; static llvm::cl::opt FormatEnum( "format", llvm::cl::desc("Format for outputted docs."), llvm::cl::values(clEnumVal(yaml, "Documentation in YAML format.")), llvm::cl::init(yaml), llvm::cl::cat(ClangDocCategory)); static llvm::cl::opt DoxygenOnly( "doxygen", llvm::cl::desc("Use only doxygen-style comments to generate docs."), llvm::cl::init(false), llvm::cl::cat(ClangDocCategory)); bool CreateDirectory(const Twine &DirName, bool ClearDirectory = false) { std::error_code OK; llvm::SmallString<128> DocsRootPath; if (ClearDirectory) { std::error_code RemoveStatus = llvm::sys::fs::remove_directories(DirName); if (RemoveStatus != OK) { llvm::errs() << "Unable to remove existing documentation directory for " << DirName << ".\n"; return true; } } std::error_code DirectoryStatus = llvm::sys::fs::create_directories(DirName); if (DirectoryStatus != OK) { llvm::errs() << "Unable to create documentation directories.\n"; return true; } return false; } bool DumpResultToFile(const Twine &DirName, const Twine &FileName, StringRef Buffer, bool ClearDirectory = false) { std::error_code OK; llvm::SmallString<128> IRRootPath; llvm::sys::path::native(OutDirectory, IRRootPath); llvm::sys::path::append(IRRootPath, DirName); if (CreateDirectory(IRRootPath, ClearDirectory)) return true; llvm::sys::path::append(IRRootPath, FileName); std::error_code OutErrorInfo; llvm::raw_fd_ostream OS(IRRootPath, OutErrorInfo, llvm::sys::fs::F_None); if (OutErrorInfo != OK) { llvm::errs() << "Error opening documentation file.\n"; return true; } OS << Buffer; OS.close(); return false; } llvm::Expected> getPath(StringRef Root, StringRef Ext, StringRef Name, llvm::SmallVectorImpl &Namespaces) { std::error_code OK; llvm::SmallString<128> Path; llvm::sys::path::native(Root, Path); for (auto R = Namespaces.rbegin(), E = Namespaces.rend(); R != E; ++R) llvm::sys::path::append(Path, R->Name); if (CreateDirectory(Path)) return llvm::make_error("Unable to create directory.\n", llvm::inconvertibleErrorCode()); llvm::sys::path::append(Path, Name + Ext); return Path; } std::string getFormatString(OutputFormatTy Ty) { switch (Ty) { case yaml: return "yaml"; } llvm_unreachable("Unknown OutputFormatTy"); } int main(int argc, const char **argv) { llvm::sys::PrintStackTraceOnErrorSignal(argv[0]); std::error_code OK; // Fail early if an invalid format was provided. std::string Format = getFormatString(FormatEnum); auto G = doc::findGeneratorByName(Format); if (!G) { llvm::errs() << toString(G.takeError()) << "\n"; return 1; } auto Exec = clang::tooling::createExecutorFromCommandLineArgs( argc, argv, ClangDocCategory); if (!Exec) { llvm::errs() << toString(Exec.takeError()) << "\n"; return 1; } ArgumentsAdjuster ArgAdjuster; if (!DoxygenOnly) ArgAdjuster = combineAdjusters( getInsertArgumentAdjuster("-fparse-all-comments", tooling::ArgumentInsertPosition::END), ArgAdjuster); // Mapping phase llvm::outs() << "Mapping decls...\n"; auto Err = Exec->get()->execute( doc::newMapperActionFactory(Exec->get()->getExecutionContext()), ArgAdjuster); if (Err) { llvm::errs() << toString(std::move(Err)) << "\n"; return 1; } if (DumpMapperResult) { bool Err = false; Exec->get()->getToolResults()->forEachResult( [&](StringRef Key, StringRef Value) { Err = DumpResultToFile("bc", Key + ".bc", Value); }); if (Err) llvm::errs() << "Error dumping map results.\n"; return Err; } // Collect values into output by key. llvm::outs() << "Collecting infos...\n"; llvm::StringMap>> MapOutput; // In ToolResults, the Key is the hashed USR and the value is the // bitcode-encoded representation of the Info object. Exec->get()->getToolResults()->forEachResult([&](StringRef Key, StringRef Value) { llvm::BitstreamCursor Stream(Value); doc::ClangDocBitcodeReader Reader(Stream); auto Infos = Reader.readBitcode(); for (auto &I : Infos) { auto R = MapOutput.try_emplace(Key, std::vector>()); R.first->second.emplace_back(std::move(I)); } }); // Reducing and generation phases llvm::outs() << "Reducing " << MapOutput.size() << " infos...\n"; llvm::StringMap> ReduceOutput; for (auto &Group : MapOutput) { auto Reduced = doc::mergeInfos(Group.getValue()); if (!Reduced) llvm::errs() << llvm::toString(Reduced.takeError()); if (DumpIntermediateResult) { SmallString<4096> Buffer; llvm::BitstreamWriter Stream(Buffer); doc::ClangDocBitcodeWriter Writer(Stream); Writer.dispatchInfoForWrite(Reduced.get().get()); if (DumpResultToFile("bc", Group.getKey() + ".bc", Buffer)) llvm::errs() << "Error dumping to bitcode.\n"; continue; } // Create the relevant ostream and emit the documentation for this decl. doc::Info *I = Reduced.get().get(); auto InfoPath = getPath(OutDirectory, "." + Format, I->Name, I->Namespace); if (!InfoPath) { llvm::errs() << toString(InfoPath.takeError()) << "\n"; continue; } std::error_code FileErr; llvm::raw_fd_ostream InfoOS(InfoPath.get(), FileErr, llvm::sys::fs::F_None); if (FileErr != OK) { llvm::errs() << "Error opening index file: " << FileErr.message() << "\n"; continue; } if (G->get()->generateDocForInfo(I, InfoOS)) llvm::errs() << "Unable to generate docs for info.\n"; } return 0; }