123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- //===- 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 <system_error>
- 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<const char *> 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<std::string> 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<std::unique_ptr<MemoryBuffer>> ManifestOrErr =
- MemoryBuffer::getFile(File);
- if (!ManifestOrErr)
- reportError(File, ManifestOrErr.getError());
- MemoryBuffer &Manifest = *ManifestOrErr.get();
- error(Merger.merge(Manifest));
- }
- std::unique_ptr<MemoryBuffer> OutputBuffer = Merger.getMergedManifest();
- if (!OutputBuffer)
- reportError("empty manifest not written");
- Expected<std::unique_ptr<FileOutputBuffer>> FileOrErr =
- FileOutputBuffer::create(OutputFile, OutputBuffer->getBufferSize());
- if (!FileOrErr)
- reportError(OutputFile, errorToErrorCode(FileOrErr.takeError()));
- std::unique_ptr<FileOutputBuffer> FileBuffer = std::move(*FileOrErr);
- std::copy(OutputBuffer->getBufferStart(), OutputBuffer->getBufferEnd(),
- FileBuffer->getBufferStart());
- error(FileBuffer->commit());
- return 0;
- }
|