1//===- llvm-mt.cpp - Merge .manifest files ---------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===---------------------------------------------------------------------===//
8//
9// Merge .manifest files. This is intended to be a platform-independent port
10// of Microsoft's mt.exe.
11//
12//===---------------------------------------------------------------------===//
13
14#include "llvm/Option/Arg.h"
15#include "llvm/Option/ArgList.h"
16#include "llvm/Option/Option.h"
17#include "llvm/Support/Error.h"
18#include "llvm/Support/FileOutputBuffer.h"
19#include "llvm/Support/LLVMDriver.h"
20#include "llvm/Support/MemoryBuffer.h"
21#include "llvm/Support/Path.h"
22#include "llvm/Support/PrettyStackTrace.h"
23#include "llvm/Support/Process.h"
24#include "llvm/Support/Signals.h"
25#include "llvm/Support/WithColor.h"
26#include "llvm/Support/raw_ostream.h"
27#include "llvm/WindowsManifest/WindowsManifestMerger.h"
28
29#include <system_error>
30
31using namespace llvm;
32
33namespace {
34
35enum ID {
36 OPT_INVALID = 0, // This is not an option ID.
37#define OPTION(...) LLVM_MAKE_OPT_ID(__VA_ARGS__),
38#include "Opts.inc"
39#undef OPTION
40};
41
42#define PREFIX(NAME, VALUE) \
43 static constexpr StringLiteral NAME##_init[] = VALUE; \
44 static constexpr ArrayRef<StringLiteral> NAME(NAME##_init, \
45 std::size(NAME##_init) - 1);
46#include "Opts.inc"
47#undef PREFIX
48
49using namespace llvm::opt;
50static constexpr opt::OptTable::Info InfoTable[] = {
51#define OPTION(...) LLVM_CONSTRUCT_OPT_INFO(__VA_ARGS__),
52#include "Opts.inc"
53#undef OPTION
54};
55
56class CvtResOptTable : public opt::GenericOptTable {
57public:
58 CvtResOptTable() : opt::GenericOptTable(InfoTable, true) {}
59};
60} // namespace
61
62[[noreturn]] static void reportError(Twine Msg) {
63 WithColor::error(OS&: errs(), Prefix: "llvm-mt") << Msg << '\n';
64 exit(status: 1);
65}
66
67static void reportError(StringRef Input, std::error_code EC) {
68 reportError(Msg: Twine(Input) + ": " + EC.message());
69}
70
71static void error(Error EC) {
72 if (EC)
73 handleAllErrors(E: std::move(EC), Handlers: [&](const ErrorInfoBase &EI) {
74 reportError(Msg: EI.message());
75 });
76}
77
78int llvm_mt_main(int Argc, char **Argv, const llvm::ToolContext &) {
79 CvtResOptTable T;
80 unsigned MAI, MAC;
81 ArrayRef<const char *> ArgsArr = ArrayRef(Argv + 1, Argc - 1);
82 opt::InputArgList InputArgs = T.ParseArgs(Args: ArgsArr, MissingArgIndex&: MAI, MissingArgCount&: MAC);
83
84 for (auto *Arg : InputArgs.filtered(Ids: OPT_INPUT)) {
85 auto ArgString = Arg->getAsString(Args: InputArgs);
86 std::string Diag;
87 raw_string_ostream OS(Diag);
88 OS << "invalid option '" << ArgString << "'";
89
90 std::string Nearest;
91 if (T.findNearest(Option: ArgString, NearestString&: Nearest) < 2)
92 OS << ", did you mean '" << Nearest << "'?";
93
94 reportError(Msg: OS.str());
95 }
96
97 for (auto &Arg : InputArgs) {
98 if (Arg->getOption().matches(ID: OPT_unsupported)) {
99 outs() << "llvm-mt: ignoring unsupported '" << Arg->getOption().getName()
100 << "' option\n";
101 }
102 }
103
104 if (InputArgs.hasArg(Ids: OPT_help)) {
105 T.printHelp(OS&: outs(), Usage: "llvm-mt [options] file...", Title: "Manifest Tool", ShowHidden: false);
106 return 0;
107 }
108
109 std::vector<std::string> InputFiles = InputArgs.getAllArgValues(Id: OPT_manifest);
110
111 if (InputFiles.size() == 0) {
112 reportError(Msg: "no input file specified");
113 }
114
115 StringRef OutputFile;
116 if (InputArgs.hasArg(Ids: OPT_out)) {
117 OutputFile = InputArgs.getLastArgValue(Id: OPT_out);
118 } else if (InputFiles.size() == 1) {
119 OutputFile = InputFiles[0];
120 } else {
121 reportError(Msg: "no output file specified");
122 }
123
124 windows_manifest::WindowsManifestMerger Merger;
125
126 for (const auto &File : InputFiles) {
127 ErrorOr<std::unique_ptr<MemoryBuffer>> ManifestOrErr =
128 MemoryBuffer::getFile(Filename: File);
129 if (!ManifestOrErr)
130 reportError(Input: File, EC: ManifestOrErr.getError());
131 error(EC: Merger.merge(Manifest: *ManifestOrErr.get()));
132 }
133
134 std::unique_ptr<MemoryBuffer> OutputBuffer = Merger.getMergedManifest();
135 if (!OutputBuffer)
136 reportError(Msg: "empty manifest not written");
137
138 int ExitCode = 0;
139 if (InputArgs.hasArg(Ids: OPT_notify_update)) {
140 ErrorOr<std::unique_ptr<MemoryBuffer>> OutBuffOrErr =
141 MemoryBuffer::getFile(Filename: OutputFile);
142 // Assume if we couldn't open the output file then it doesn't exist meaning
143 // there was a change.
144 bool Same = false;
145 if (OutBuffOrErr) {
146 const std::unique_ptr<MemoryBuffer> &FileBuffer = *OutBuffOrErr;
147 Same = std::equal(
148 first1: OutputBuffer->getBufferStart(), last1: OutputBuffer->getBufferEnd(),
149 first2: FileBuffer->getBufferStart(), last2: FileBuffer->getBufferEnd());
150 }
151 if (!Same) {
152#if LLVM_ON_UNIX
153 ExitCode = 0xbb;
154#elif defined(_WIN32)
155 ExitCode = 0x41020001;
156#endif
157 }
158 }
159
160 Expected<std::unique_ptr<FileOutputBuffer>> FileOrErr =
161 FileOutputBuffer::create(FilePath: OutputFile, Size: OutputBuffer->getBufferSize());
162 if (!FileOrErr)
163 reportError(Input: OutputFile, EC: errorToErrorCode(Err: FileOrErr.takeError()));
164 std::unique_ptr<FileOutputBuffer> FileBuffer = std::move(*FileOrErr);
165 std::copy(first: OutputBuffer->getBufferStart(), last: OutputBuffer->getBufferEnd(),
166 result: FileBuffer->getBufferStart());
167 error(EC: FileBuffer->commit());
168 return ExitCode;
169}
170