//===- llvm-mt.cpp - Merge .manifest files ---------------------*- 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 // //===---------------------------------------------------------------------===// // // Merge .manifest files. This is intended to be a platform-independent port // of Microsoft's mt.exe. // //===---------------------------------------------------------------------===// #include "llvm/Option/Arg.h" #include "llvm/Option/ArgList.h" #include "llvm/Option/Option.h" #include "llvm/Support/Error.h" #include "llvm/Support/FileOutputBuffer.h" #include "llvm/Support/InitLLVM.h" #include "llvm/Support/ManagedStatic.h" #include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/Path.h" #include "llvm/Support/PrettyStackTrace.h" #include "llvm/Support/Process.h" #include "llvm/Support/Signals.h" #include "llvm/Support/WithColor.h" #include "llvm/Support/raw_ostream.h" #include "llvm/WindowsManifest/WindowsManifestMerger.h" #include using namespace llvm; namespace { enum ID { OPT_INVALID = 0, // This is not an option ID. #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ HELPTEXT, METAVAR, VALUES) \ OPT_##ID, #include "Opts.inc" #undef OPTION }; #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE; #include "Opts.inc" #undef PREFIX static const opt::OptTable::Info InfoTable[] = { #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ HELPTEXT, METAVAR, VALUES) \ { \ PREFIX, NAME, HELPTEXT, \ METAVAR, OPT_##ID, opt::Option::KIND##Class, \ PARAM, FLAGS, OPT_##GROUP, \ OPT_##ALIAS, ALIASARGS, VALUES}, #include "Opts.inc" #undef OPTION }; class CvtResOptTable : public opt::OptTable { public: CvtResOptTable() : OptTable(InfoTable, true) {} }; } // namespace LLVM_ATTRIBUTE_NORETURN static void reportError(Twine Msg) { WithColor::error(errs(), "llvm-mt") << Msg << '\n'; exit(1); } static void reportError(StringRef Input, std::error_code EC) { reportError(Twine(Input) + ": " + EC.message()); } static void error(Error EC) { if (EC) handleAllErrors(std::move(EC), [&](const ErrorInfoBase &EI) { reportError(EI.message()); }); } int main(int Argc, const char **Argv) { InitLLVM X(Argc, Argv); CvtResOptTable T; unsigned MAI, MAC; ArrayRef ArgsArr = makeArrayRef(Argv + 1, Argc - 1); opt::InputArgList InputArgs = T.ParseArgs(ArgsArr, MAI, MAC); for (auto *Arg : InputArgs.filtered(OPT_INPUT)) { auto ArgString = Arg->getAsString(InputArgs); std::string Diag; raw_string_ostream OS(Diag); OS << "invalid option '" << ArgString << "'"; std::string Nearest; if (T.findNearest(ArgString, Nearest) < 2) OS << ", did you mean '" << Nearest << "'?"; reportError(OS.str()); } for (auto &Arg : InputArgs) { if (Arg->getOption().matches(OPT_unsupported)) { outs() << "llvm-mt: ignoring unsupported '" << Arg->getOption().getName() << "' option\n"; } } if (InputArgs.hasArg(OPT_help)) { T.PrintHelp(outs(), "llvm-mt [options] file...", "Manifest Tool", false); return 0; } std::vector InputFiles = InputArgs.getAllArgValues(OPT_manifest); if (InputFiles.size() == 0) { reportError("no input file specified"); } StringRef OutputFile; if (InputArgs.hasArg(OPT_out)) { OutputFile = InputArgs.getLastArgValue(OPT_out); } else if (InputFiles.size() == 1) { OutputFile = InputFiles[0]; } else { reportError("no output file specified"); } windows_manifest::WindowsManifestMerger Merger; for (const auto &File : InputFiles) { ErrorOr> ManifestOrErr = MemoryBuffer::getFile(File); if (!ManifestOrErr) reportError(File, ManifestOrErr.getError()); MemoryBuffer &Manifest = *ManifestOrErr.get(); error(Merger.merge(Manifest)); } std::unique_ptr OutputBuffer = Merger.getMergedManifest(); if (!OutputBuffer) reportError("empty manifest not written"); Expected> FileOrErr = FileOutputBuffer::create(OutputFile, OutputBuffer->getBufferSize()); if (!FileOrErr) reportError(OutputFile, errorToErrorCode(FileOrErr.takeError())); std::unique_ptr FileBuffer = std::move(*FileOrErr); std::copy(OutputBuffer->getBufferStart(), OutputBuffer->getBufferEnd(), FileBuffer->getBufferStart()); error(FileBuffer->commit()); return 0; }