|  | //===-- cc1gen_reproducer_main.cpp - Clang reproducer generator  ----------===// | 
|  | // | 
|  | // 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 | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This is the entry point to the clang -cc1gen-reproducer functionality, which | 
|  | // generates reproducers for invocations for clang-based tools. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "clang/Basic/Diagnostic.h" | 
|  | #include "clang/Basic/LLVM.h" | 
|  | #include "clang/Driver/Compilation.h" | 
|  | #include "clang/Driver/Driver.h" | 
|  | #include "llvm/ADT/ArrayRef.h" | 
|  | #include "llvm/ADT/STLExtras.h" | 
|  | #include "llvm/Support/FileSystem.h" | 
|  | #include "llvm/Support/LLVMDriver.h" | 
|  | #include "llvm/Support/TargetSelect.h" | 
|  | #include "llvm/Support/VirtualFileSystem.h" | 
|  | #include "llvm/Support/YAMLTraits.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  | #include "llvm/TargetParser/Host.h" | 
|  | #include <optional> | 
|  |  | 
|  | using namespace clang; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | struct UnsavedFileHash { | 
|  | std::string Name; | 
|  | std::string MD5; | 
|  | }; | 
|  |  | 
|  | struct ClangInvocationInfo { | 
|  | std::string Toolchain; | 
|  | std::string LibclangOperation; | 
|  | std::string LibclangOptions; | 
|  | std::vector<std::string> Arguments; | 
|  | std::vector<std::string> InvocationArguments; | 
|  | std::vector<UnsavedFileHash> UnsavedFileHashes; | 
|  | bool Dump = false; | 
|  | }; | 
|  |  | 
|  | } // end anonymous namespace | 
|  |  | 
|  | LLVM_YAML_IS_SEQUENCE_VECTOR(UnsavedFileHash) | 
|  |  | 
|  | namespace llvm { | 
|  | namespace yaml { | 
|  |  | 
|  | template <> struct MappingTraits<UnsavedFileHash> { | 
|  | static void mapping(IO &IO, UnsavedFileHash &Info) { | 
|  | IO.mapRequired("name", Info.Name); | 
|  | IO.mapRequired("md5", Info.MD5); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct MappingTraits<ClangInvocationInfo> { | 
|  | static void mapping(IO &IO, ClangInvocationInfo &Info) { | 
|  | IO.mapRequired("toolchain", Info.Toolchain); | 
|  | IO.mapOptional("libclang.operation", Info.LibclangOperation); | 
|  | IO.mapOptional("libclang.opts", Info.LibclangOptions); | 
|  | IO.mapRequired("args", Info.Arguments); | 
|  | IO.mapOptional("invocation-args", Info.InvocationArguments); | 
|  | IO.mapOptional("unsaved_file_hashes", Info.UnsavedFileHashes); | 
|  | } | 
|  | }; | 
|  |  | 
|  | } // end namespace yaml | 
|  | } // end namespace llvm | 
|  |  | 
|  | static std::string generateReproducerMetaInfo(const ClangInvocationInfo &Info) { | 
|  | std::string Result; | 
|  | llvm::raw_string_ostream OS(Result); | 
|  | OS << '{'; | 
|  | bool NeedComma = false; | 
|  | auto EmitKey = [&](StringRef Key) { | 
|  | if (NeedComma) | 
|  | OS << ", "; | 
|  | NeedComma = true; | 
|  | OS << '"' << Key << "\": "; | 
|  | }; | 
|  | auto EmitStringKey = [&](StringRef Key, StringRef Value) { | 
|  | if (Value.empty()) | 
|  | return; | 
|  | EmitKey(Key); | 
|  | OS << '"' << Value << '"'; | 
|  | }; | 
|  | EmitStringKey("libclang.operation", Info.LibclangOperation); | 
|  | EmitStringKey("libclang.opts", Info.LibclangOptions); | 
|  | if (!Info.InvocationArguments.empty()) { | 
|  | EmitKey("invocation-args"); | 
|  | OS << '['; | 
|  | for (const auto &Arg : llvm::enumerate(Info.InvocationArguments)) { | 
|  | if (Arg.index()) | 
|  | OS << ','; | 
|  | OS << '"' << Arg.value() << '"'; | 
|  | } | 
|  | OS << ']'; | 
|  | } | 
|  | OS << '}'; | 
|  | // FIXME: Compare unsaved file hashes and report mismatch in the reproducer. | 
|  | if (Info.Dump) | 
|  | llvm::outs() << "REPRODUCER METAINFO: " << Result << "\n"; | 
|  | return Result; | 
|  | } | 
|  |  | 
|  | /// Generates a reproducer for a set of arguments from a specific invocation. | 
|  | static std::optional<driver::Driver::CompilationDiagnosticReport> | 
|  | generateReproducerForInvocationArguments(ArrayRef<const char *> Argv, | 
|  | const ClangInvocationInfo &Info, | 
|  | const llvm::ToolContext &ToolContext) { | 
|  | using namespace driver; | 
|  | auto TargetAndMode = ToolChain::getTargetAndModeFromProgramName(Argv[0]); | 
|  |  | 
|  | DiagnosticOptions DiagOpts; | 
|  |  | 
|  | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); | 
|  | DiagnosticsEngine Diags(DiagID, DiagOpts, new IgnoringDiagConsumer()); | 
|  | auto VFS = llvm::vfs::getRealFileSystem(); | 
|  | ProcessWarningOptions(Diags, DiagOpts, *VFS, /*ReportDiags=*/false); | 
|  | Driver TheDriver(ToolContext.Path, llvm::sys::getDefaultTargetTriple(), Diags, | 
|  | /*Title=*/"clang LLVM compiler", VFS); | 
|  | TheDriver.setTargetAndMode(TargetAndMode); | 
|  | if (ToolContext.NeedsPrependArg) | 
|  | TheDriver.setPrependArg(ToolContext.PrependArg); | 
|  |  | 
|  | std::unique_ptr<Compilation> C(TheDriver.BuildCompilation(Argv)); | 
|  | if (C && !C->containsError()) { | 
|  | for (const auto &J : C->getJobs()) { | 
|  | if (const Command *Cmd = dyn_cast<Command>(&J)) { | 
|  | Driver::CompilationDiagnosticReport Report; | 
|  | TheDriver.generateCompilationDiagnostics( | 
|  | *C, *Cmd, generateReproducerMetaInfo(Info), &Report); | 
|  | return Report; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return std::nullopt; | 
|  | } | 
|  |  | 
|  | std::string GetExecutablePath(const char *Argv0, bool CanonicalPrefixes); | 
|  |  | 
|  | static void printReproducerInformation( | 
|  | llvm::raw_ostream &OS, const ClangInvocationInfo &Info, | 
|  | const driver::Driver::CompilationDiagnosticReport &Report) { | 
|  | OS << "REPRODUCER:\n"; | 
|  | OS << "{\n"; | 
|  | OS << R"("files":[)"; | 
|  | for (const auto &File : llvm::enumerate(Report.TemporaryFiles)) { | 
|  | if (File.index()) | 
|  | OS << ','; | 
|  | OS << '"' << File.value() << '"'; | 
|  | } | 
|  | OS << "]\n}\n"; | 
|  | } | 
|  |  | 
|  | int cc1gen_reproducer_main(ArrayRef<const char *> Argv, const char *Argv0, | 
|  | void *MainAddr, | 
|  | const llvm::ToolContext &ToolContext) { | 
|  | if (Argv.size() < 1) { | 
|  | llvm::errs() << "error: missing invocation file\n"; | 
|  | return 1; | 
|  | } | 
|  | // Parse the invocation descriptor. | 
|  | StringRef Input = Argv[0]; | 
|  | llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Buffer = | 
|  | llvm::MemoryBuffer::getFile(Input, /*IsText=*/true); | 
|  | if (!Buffer) { | 
|  | llvm::errs() << "error: failed to read " << Input << ": " | 
|  | << Buffer.getError().message() << "\n"; | 
|  | return 1; | 
|  | } | 
|  | llvm::yaml::Input YAML(Buffer.get()->getBuffer()); | 
|  | ClangInvocationInfo InvocationInfo; | 
|  | YAML >> InvocationInfo; | 
|  | if (Argv.size() > 1 && Argv[1] == StringRef("-v")) | 
|  | InvocationInfo.Dump = true; | 
|  |  | 
|  | // Create an invocation that will produce the reproducer. | 
|  | std::vector<const char *> DriverArgs; | 
|  | for (const auto &Arg : InvocationInfo.Arguments) | 
|  | DriverArgs.push_back(Arg.c_str()); | 
|  | std::string Path = GetExecutablePath(Argv0, /*CanonicalPrefixes=*/true); | 
|  | DriverArgs[0] = Path.c_str(); | 
|  | std::optional<driver::Driver::CompilationDiagnosticReport> Report = | 
|  | generateReproducerForInvocationArguments(DriverArgs, InvocationInfo, | 
|  | ToolContext); | 
|  |  | 
|  | // Emit the information about the reproduce files to stdout. | 
|  | int Result = 1; | 
|  | if (Report) { | 
|  | printReproducerInformation(llvm::outs(), InvocationInfo, *Report); | 
|  | Result = 0; | 
|  | } | 
|  |  | 
|  | // Remove the input file. | 
|  | llvm::sys::fs::remove(Input); | 
|  | return Result; | 
|  | } |