1 //===- llvm-cvtres.cpp - Serialize .res files into .obj ---------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 // Serialize .res files into .obj files. This is intended to be a
11 // platform-independent port of Microsoft's cvtres.exe.
13 //===----------------------------------------------------------------------===//
15 #include "llvm/ADT/StringSwitch.h"
16 #include "llvm/Object/Binary.h"
17 #include "llvm/Object/WindowsResource.h"
18 #include "llvm/Option/Arg.h"
19 #include "llvm/Option/ArgList.h"
20 #include "llvm/Option/Option.h"
21 #include "llvm/Support/BinaryStreamError.h"
22 #include "llvm/Support/Error.h"
23 #include "llvm/Support/InitLLVM.h"
24 #include "llvm/Support/ManagedStatic.h"
25 #include "llvm/Support/Path.h"
26 #include "llvm/Support/PrettyStackTrace.h"
27 #include "llvm/Support/Process.h"
28 #include "llvm/Support/Signals.h"
29 #include "llvm/Support/raw_ostream.h"
31 #include <system_error>
34 using namespace object;
39 OPT_INVALID = 0, // This is not an option ID.
40 #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
41 HELPTEXT, METAVAR, VALUES) \
47 #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
51 static const opt::OptTable::Info InfoTable[] = {
52 #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
53 HELPTEXT, METAVAR, VALUES) \
55 PREFIX, NAME, HELPTEXT, \
56 METAVAR, OPT_##ID, opt::Option::KIND##Class, \
57 PARAM, FLAGS, OPT_##GROUP, \
58 OPT_##ALIAS, ALIASARGS, VALUES},
63 class CvtResOptTable : public opt::OptTable {
65 CvtResOptTable() : OptTable(InfoTable, true) {}
69 LLVM_ATTRIBUTE_NORETURN void reportError(Twine Msg) {
74 static void reportError(StringRef Input, std::error_code EC) {
75 reportError(Twine(Input) + ": " + EC.message() + ".\n");
78 void error(std::error_code EC) {
81 reportError(EC.message() + ".\n");
84 void error(Error EC) {
87 handleAllErrors(std::move(EC),
88 [&](const ErrorInfoBase &EI) { reportError(EI.message()); });
91 template <typename T> T error(Expected<T> EC) {
93 error(EC.takeError());
94 return std::move(EC.get());
97 int main(int Argc, const char **Argv) {
98 InitLLVM X(Argc, Argv);
102 ArrayRef<const char *> ArgsArr = makeArrayRef(Argv + 1, Argc - 1);
103 opt::InputArgList InputArgs = T.ParseArgs(ArgsArr, MAI, MAC);
105 if (InputArgs.hasArg(OPT_HELP)) {
106 T.PrintHelp(outs(), "cvtres", "Resource Converter", false);
110 bool Verbose = InputArgs.hasArg(OPT_VERBOSE);
112 COFF::MachineTypes MachineType;
114 if (InputArgs.hasArg(OPT_MACHINE)) {
115 std::string MachineString = InputArgs.getLastArgValue(OPT_MACHINE).upper();
116 MachineType = StringSwitch<COFF::MachineTypes>(MachineString)
117 .Case("ARM", COFF::IMAGE_FILE_MACHINE_ARMNT)
118 .Case("ARM64", COFF::IMAGE_FILE_MACHINE_ARM64)
119 .Case("X64", COFF::IMAGE_FILE_MACHINE_AMD64)
120 .Case("X86", COFF::IMAGE_FILE_MACHINE_I386)
121 .Default(COFF::IMAGE_FILE_MACHINE_UNKNOWN);
122 if (MachineType == COFF::IMAGE_FILE_MACHINE_UNKNOWN)
123 reportError("Unsupported machine architecture");
126 outs() << "Machine architecture not specified; assumed X64.\n";
127 MachineType = COFF::IMAGE_FILE_MACHINE_AMD64;
130 std::vector<std::string> InputFiles = InputArgs.getAllArgValues(OPT_INPUT);
132 if (InputFiles.size() == 0) {
133 reportError("No input file specified.\n");
136 SmallString<128> OutputFile;
138 if (InputArgs.hasArg(OPT_OUT)) {
139 OutputFile = InputArgs.getLastArgValue(OPT_OUT);
141 OutputFile = sys::path::filename(StringRef(InputFiles[0]));
142 sys::path::replace_extension(OutputFile, ".obj");
146 outs() << "Machine: ";
147 switch (MachineType) {
148 case COFF::IMAGE_FILE_MACHINE_ARM64:
151 case COFF::IMAGE_FILE_MACHINE_ARMNT:
154 case COFF::IMAGE_FILE_MACHINE_I386:
162 WindowsResourceParser Parser;
164 for (const auto &File : InputFiles) {
165 Expected<OwningBinary<Binary>> BinaryOrErr = createBinary(File);
167 reportError(File, errorToErrorCode(BinaryOrErr.takeError()));
169 Binary &Binary = *BinaryOrErr.get().getBinary();
171 WindowsResource *RF = dyn_cast<WindowsResource>(&Binary);
173 reportError(File + ": unrecognized file format.\n");
177 ResourceEntryRef Entry = error(RF->getHeadEntry());
180 error(Entry.moveNext(End));
183 outs() << "Number of resources: " << EntryNumber << "\n";
186 error(Parser.parse(RF));
190 Parser.printTree(outs());
193 std::unique_ptr<MemoryBuffer> OutputBuffer =
194 error(llvm::object::writeWindowsResourceCOFF(MachineType, Parser));
196 FileOutputBuffer::create(OutputFile, OutputBuffer->getBufferSize());
198 reportError(OutputFile, errorToErrorCode(FileOrErr.takeError()));
199 std::unique_ptr<FileOutputBuffer> FileBuffer = std::move(*FileOrErr);
200 std::copy(OutputBuffer->getBufferStart(), OutputBuffer->getBufferEnd(),
201 FileBuffer->getBufferStart());
202 error(FileBuffer->commit());
205 Expected<OwningBinary<Binary>> BinaryOrErr = createBinary(OutputFile);
207 reportError(OutputFile, errorToErrorCode(BinaryOrErr.takeError()));
208 Binary &Binary = *BinaryOrErr.get().getBinary();
209 ScopedPrinter W(errs());
210 W.printBinaryBlock("Output File Raw Data", Binary.getData());