blob: 53989c8191faaf710f19841e87d22a54e044fde0 [file] [log] [blame]
Zachary Turner83e112a2017-04-21 17:30:29 +00001//===- llvm-cvtres.cpp - Serialize .res files into .obj ---------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// Serialize .res files into .obj files. This is intended to be a
11// platform-independent port of Microsoft's cvtres.exe.
12//
13//===----------------------------------------------------------------------===//
14
Eric Beckmann836dd8e2017-05-20 01:49:19 +000015#include "llvm/ADT/StringSwitch.h"
16#include "llvm/Object/Binary.h"
17#include "llvm/Object/WindowsResource.h"
Zachary Turner83e112a2017-04-21 17:30:29 +000018#include "llvm/Option/Arg.h"
19#include "llvm/Option/ArgList.h"
20#include "llvm/Option/Option.h"
Eric Beckmann836dd8e2017-05-20 01:49:19 +000021#include "llvm/Support/BinaryStreamError.h"
Zachary Turner83e112a2017-04-21 17:30:29 +000022#include "llvm/Support/Error.h"
Rui Ueyama0b9d56a2018-04-13 18:26:06 +000023#include "llvm/Support/InitLLVM.h"
Zachary Turner83e112a2017-04-21 17:30:29 +000024#include "llvm/Support/ManagedStatic.h"
Eric Beckmann836dd8e2017-05-20 01:49:19 +000025#include "llvm/Support/Path.h"
Zachary Turner83e112a2017-04-21 17:30:29 +000026#include "llvm/Support/PrettyStackTrace.h"
27#include "llvm/Support/Process.h"
28#include "llvm/Support/Signals.h"
29#include "llvm/Support/raw_ostream.h"
30
Eric Beckmann51c5f772017-06-16 22:00:42 +000031#include <system_error>
32
Zachary Turner83e112a2017-04-21 17:30:29 +000033using namespace llvm;
Eric Beckmann836dd8e2017-05-20 01:49:19 +000034using namespace object;
Zachary Turner83e112a2017-04-21 17:30:29 +000035
36namespace {
37
38enum ID {
39 OPT_INVALID = 0, // This is not an option ID.
40#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
Yuka Takahashibc5df292017-06-20 16:31:31 +000041 HELPTEXT, METAVAR, VALUES) \
Zachary Turner83e112a2017-04-21 17:30:29 +000042 OPT_##ID,
43#include "Opts.inc"
44#undef OPTION
45};
46
47#define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
48#include "Opts.inc"
49#undef PREFIX
50
51static const opt::OptTable::Info InfoTable[] = {
52#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
Yuka Takahashibc5df292017-06-20 16:31:31 +000053 HELPTEXT, METAVAR, VALUES) \
Zachary Turner83e112a2017-04-21 17:30:29 +000054 { \
Yuka Takahashibc5df292017-06-20 16:31:31 +000055 PREFIX, NAME, HELPTEXT, \
56 METAVAR, OPT_##ID, opt::Option::KIND##Class, \
57 PARAM, FLAGS, OPT_##GROUP, \
58 OPT_##ALIAS, ALIASARGS, VALUES},
Zachary Turner83e112a2017-04-21 17:30:29 +000059#include "Opts.inc"
60#undef OPTION
61};
62
63class CvtResOptTable : public opt::OptTable {
64public:
65 CvtResOptTable() : OptTable(InfoTable, true) {}
66};
Zachary Turner83e112a2017-04-21 17:30:29 +000067}
68
Eric Beckmann836dd8e2017-05-20 01:49:19 +000069LLVM_ATTRIBUTE_NORETURN void reportError(Twine Msg) {
70 errs() << Msg;
71 exit(1);
72}
73
74static void reportError(StringRef Input, std::error_code EC) {
75 reportError(Twine(Input) + ": " + EC.message() + ".\n");
76}
77
78void error(std::error_code EC) {
79 if (!EC)
80 return;
81 reportError(EC.message() + ".\n");
82}
83
84void error(Error EC) {
85 if (!EC)
86 return;
87 handleAllErrors(std::move(EC),
88 [&](const ErrorInfoBase &EI) { reportError(EI.message()); });
89}
90
Eric Beckmannb741c532017-06-19 18:49:05 +000091template <typename T> T error(Expected<T> EC) {
92 if (!EC)
93 error(EC.takeError());
94 return std::move(EC.get());
95}
96
Rui Ueyama0b9d56a2018-04-13 18:26:06 +000097int main(int Argc, const char **Argv) {
98 InitLLVM X(Argc, Argv);
Zachary Turner83e112a2017-04-21 17:30:29 +000099
100 CvtResOptTable T;
101 unsigned MAI, MAC;
Rui Ueyama0b9d56a2018-04-13 18:26:06 +0000102 ArrayRef<const char *> ArgsArr = makeArrayRef(Argv + 1, Argc - 1);
Zachary Turner83e112a2017-04-21 17:30:29 +0000103 opt::InputArgList InputArgs = T.ParseArgs(ArgsArr, MAI, MAC);
104
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000105 if (InputArgs.hasArg(OPT_HELP)) {
Fangrui Song42f63b62018-10-10 00:15:31 +0000106 T.PrintHelp(outs(), "llvm-cvtres [options] file...", "Resource Converter", false);
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000107 return 0;
108 }
Zachary Turner83e112a2017-04-21 17:30:29 +0000109
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000110 bool Verbose = InputArgs.hasArg(OPT_VERBOSE);
111
Eric Beckmann1efceea2017-07-05 19:04:33 +0000112 COFF::MachineTypes MachineType;
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000113
114 if (InputArgs.hasArg(OPT_MACHINE)) {
115 std::string MachineString = InputArgs.getLastArgValue(OPT_MACHINE).upper();
Eric Beckmann1efceea2017-07-05 19:04:33 +0000116 MachineType = StringSwitch<COFF::MachineTypes>(MachineString)
117 .Case("ARM", COFF::IMAGE_FILE_MACHINE_ARMNT)
Martin Storsjo4a22e742017-11-10 22:27:41 +0000118 .Case("ARM64", COFF::IMAGE_FILE_MACHINE_ARM64)
Eric Beckmann1efceea2017-07-05 19:04:33 +0000119 .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)
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000123 reportError("Unsupported machine architecture");
124 } else {
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000125 if (Verbose)
126 outs() << "Machine architecture not specified; assumed X64.\n";
Eric Beckmann1efceea2017-07-05 19:04:33 +0000127 MachineType = COFF::IMAGE_FILE_MACHINE_AMD64;
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000128 }
129
130 std::vector<std::string> InputFiles = InputArgs.getAllArgValues(OPT_INPUT);
131
132 if (InputFiles.size() == 0) {
Eric Beckmann1f0488c2017-05-30 18:19:06 +0000133 reportError("No input file specified.\n");
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000134 }
135
136 SmallString<128> OutputFile;
137
138 if (InputArgs.hasArg(OPT_OUT)) {
139 OutputFile = InputArgs.getLastArgValue(OPT_OUT);
140 } else {
Eric Beckmann51c5f772017-06-16 22:00:42 +0000141 OutputFile = sys::path::filename(StringRef(InputFiles[0]));
142 sys::path::replace_extension(OutputFile, ".obj");
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000143 }
144
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000145 if (Verbose) {
146 outs() << "Machine: ";
147 switch (MachineType) {
Martin Storsjo4a22e742017-11-10 22:27:41 +0000148 case COFF::IMAGE_FILE_MACHINE_ARM64:
149 outs() << "ARM64\n";
150 break;
Eric Beckmann1efceea2017-07-05 19:04:33 +0000151 case COFF::IMAGE_FILE_MACHINE_ARMNT:
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000152 outs() << "ARM\n";
153 break;
Eric Beckmann1efceea2017-07-05 19:04:33 +0000154 case COFF::IMAGE_FILE_MACHINE_I386:
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000155 outs() << "X86\n";
156 break;
157 default:
158 outs() << "X64\n";
159 }
Eric Beckmann1f0488c2017-05-30 18:19:06 +0000160 }
161
162 WindowsResourceParser Parser;
163
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000164 for (const auto &File : InputFiles) {
Eric Beckmann51c5f772017-06-16 22:00:42 +0000165 Expected<OwningBinary<Binary>> BinaryOrErr = createBinary(File);
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000166 if (!BinaryOrErr)
167 reportError(File, errorToErrorCode(BinaryOrErr.takeError()));
168
169 Binary &Binary = *BinaryOrErr.get().getBinary();
170
171 WindowsResource *RF = dyn_cast<WindowsResource>(&Binary);
172 if (!RF)
173 reportError(File + ": unrecognized file format.\n");
174
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000175 if (Verbose) {
176 int EntryNumber = 0;
Eric Beckmannb741c532017-06-19 18:49:05 +0000177 ResourceEntryRef Entry = error(RF->getHeadEntry());
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000178 bool End = false;
179 while (!End) {
180 error(Entry.moveNext(End));
181 EntryNumber++;
182 }
183 outs() << "Number of resources: " << EntryNumber << "\n";
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000184 }
Eric Beckmann1f0488c2017-05-30 18:19:06 +0000185
186 error(Parser.parse(RF));
Eric Beckmann836dd8e2017-05-20 01:49:19 +0000187 }
Eric Beckmann1f0488c2017-05-30 18:19:06 +0000188
Eric Beckmannf9d223b2017-06-13 18:17:36 +0000189 if (Verbose) {
190 Parser.printTree(outs());
Eric Beckmannf9d223b2017-06-13 18:17:36 +0000191 }
Eric Beckmann3028aaf2017-06-09 17:34:30 +0000192
Eric Beckmannb741c532017-06-19 18:49:05 +0000193 std::unique_ptr<MemoryBuffer> OutputBuffer =
194 error(llvm::object::writeWindowsResourceCOFF(MachineType, Parser));
Eric Beckmannc501cd42017-06-16 21:13:24 +0000195 auto FileOrErr =
196 FileOutputBuffer::create(OutputFile, OutputBuffer->getBufferSize());
197 if (!FileOrErr)
Rafael Espindola0fa582d2017-11-08 01:05:44 +0000198 reportError(OutputFile, errorToErrorCode(FileOrErr.takeError()));
Eric Beckmannc501cd42017-06-16 21:13:24 +0000199 std::unique_ptr<FileOutputBuffer> FileBuffer = std::move(*FileOrErr);
200 std::copy(OutputBuffer->getBufferStart(), OutputBuffer->getBufferEnd(),
201 FileBuffer->getBufferStart());
202 error(FileBuffer->commit());
Eric Beckmannb618c822017-07-08 03:06:10 +0000203
Eric Beckmannf9d223b2017-06-13 18:17:36 +0000204 if (Verbose) {
Eric Beckmann51c5f772017-06-16 22:00:42 +0000205 Expected<OwningBinary<Binary>> BinaryOrErr = createBinary(OutputFile);
Eric Beckmannf9d223b2017-06-13 18:17:36 +0000206 if (!BinaryOrErr)
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());
211 }
212
Zachary Turner83e112a2017-04-21 17:30:29 +0000213 return 0;
214}