123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155 |
- //===- ClangSrcLocDump.cpp ------------------------------------*- 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
- //
- //===----------------------------------------------------------------------===//
- #include "clang/Basic/Diagnostic.h"
- #include "clang/Driver/Compilation.h"
- #include "clang/Driver/Driver.h"
- #include "clang/Driver/Job.h"
- #include "clang/Driver/Options.h"
- #include "clang/Driver/Tool.h"
- #include "clang/Frontend/CompilerInstance.h"
- #include "clang/Frontend/TextDiagnosticPrinter.h"
- #include "clang/Lex/PreprocessorOptions.h"
- #include "clang/Tooling/Tooling.h"
- #include "llvm/Option/ArgList.h"
- #include "llvm/Support/CommandLine.h"
- #include "llvm/Support/Host.h"
- #include "llvm/Support/JSON.h"
- #include "ASTSrcLocProcessor.h"
- using namespace clang::tooling;
- using namespace clang;
- using namespace llvm;
- static cl::list<std::string> IncludeDirectories(
- "I", cl::desc("Include directories to use while compiling"),
- cl::value_desc("directory"), cl::Required, cl::OneOrMore, cl::Prefix);
- static cl::opt<bool>
- SkipProcessing("skip-processing",
- cl::desc("Avoid processing the AST header file"),
- cl::Required, cl::value_desc("bool"));
- static cl::opt<std::string> JsonOutputPath("json-output-path",
- cl::desc("json output path"),
- cl::Required,
- cl::value_desc("path"));
- class ASTSrcLocGenerationAction : public clang::ASTFrontendAction {
- public:
- ASTSrcLocGenerationAction() : Processor(JsonOutputPath) {}
- void ExecuteAction() override {
- clang::ASTFrontendAction::ExecuteAction();
- if (getCompilerInstance().getDiagnostics().getNumErrors() > 0)
- Processor.generateEmpty();
- else
- Processor.generate();
- }
- std::unique_ptr<clang::ASTConsumer>
- CreateASTConsumer(clang::CompilerInstance &Compiler,
- llvm::StringRef File) override {
- return Processor.createASTConsumer(Compiler, File);
- }
- private:
- ASTSrcLocProcessor Processor;
- };
- static const char Filename[] = "ASTTU.cpp";
- int main(int argc, const char **argv) {
- cl::ParseCommandLineOptions(argc, argv);
- if (SkipProcessing) {
- std::error_code EC;
- llvm::raw_fd_ostream JsonOut(JsonOutputPath, EC, llvm::sys::fs::OF_Text);
- if (EC)
- return 1;
- JsonOut << formatv("{0:2}", llvm::json::Value(llvm::json::Object()));
- return 0;
- }
- std::vector<std::string> Args;
- Args.push_back("-cc1");
- llvm::transform(IncludeDirectories, std::back_inserter(Args),
- [](const std::string &IncDir) { return "-I" + IncDir; });
- Args.push_back("-fsyntax-only");
- Args.push_back(Filename);
- std::vector<const char *> Argv(Args.size(), nullptr);
- llvm::transform(Args, Argv.begin(),
- [](const std::string &Arg) { return Arg.c_str(); });
- IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts =
- CreateAndPopulateDiagOpts(Argv);
- // Don't output diagnostics, because common scenarios such as
- // cross-compiling fail with diagnostics. This is not fatal, but
- // just causes attempts to use the introspection API to return no data.
- TextDiagnosticPrinter DiagnosticPrinter(llvm::nulls(), &*DiagOpts);
- DiagnosticsEngine Diagnostics(
- IntrusiveRefCntPtr<DiagnosticIDs>(new DiagnosticIDs()), &*DiagOpts,
- &DiagnosticPrinter, false);
- auto *OFS = new llvm::vfs::OverlayFileSystem(vfs::getRealFileSystem());
- auto *MemFS = new llvm::vfs::InMemoryFileSystem();
- OFS->pushOverlay(MemFS);
- MemFS->addFile(Filename, 0,
- MemoryBuffer::getMemBuffer("#include \"clang/AST/AST.h\"\n"));
- auto Files = llvm::makeIntrusiveRefCnt<FileManager>(FileSystemOptions(), OFS);
- auto Driver = std::make_unique<driver::Driver>(
- "clang", llvm::sys::getDefaultTargetTriple(), Diagnostics,
- "ast-api-dump-tool", OFS);
- std::unique_ptr<clang::driver::Compilation> Comp(
- Driver->BuildCompilation(llvm::ArrayRef(Argv)));
- if (!Comp)
- return 1;
- const auto &Jobs = Comp->getJobs();
- if (Jobs.size() != 1 || !isa<driver::Command>(*Jobs.begin())) {
- SmallString<256> error_msg;
- llvm::raw_svector_ostream error_stream(error_msg);
- Jobs.Print(error_stream, "; ", true);
- return 1;
- }
- const auto &Cmd = cast<driver::Command>(*Jobs.begin());
- const llvm::opt::ArgStringList &CC1Args = Cmd.getArguments();
- auto Invocation = std::make_unique<CompilerInvocation>();
- CompilerInvocation::CreateFromArgs(*Invocation, CC1Args, Diagnostics);
- CompilerInstance Compiler(std::make_shared<clang::PCHContainerOperations>());
- Compiler.setInvocation(std::move(Invocation));
- Compiler.createDiagnostics(&DiagnosticPrinter, false);
- if (!Compiler.hasDiagnostics())
- return 1;
- // Suppress "2 errors generated" or similar messages
- Compiler.getDiagnosticOpts().ShowCarets = false;
- Compiler.createSourceManager(*Files);
- Compiler.setFileManager(Files.get());
- ASTSrcLocGenerationAction ScopedToolAction;
- Compiler.ExecuteAction(ScopedToolAction);
- Files->clearStatCache();
- return 0;
- }
|