1 //===- llvm-cvtres.cpp - Serialize .res files into .obj ---------*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 // Serialize .res files into .obj files. This is intended to be a
10 // platform-independent port of Microsoft's cvtres.exe.
12 //===----------------------------------------------------------------------===//
14 #include "llvm/ADT/StringSwitch.h"
15 #include "llvm/Object/Binary.h"
16 #include "llvm/Object/WindowsResource.h"
17 #include "llvm/Option/Arg.h"
18 #include "llvm/Option/ArgList.h"
19 #include "llvm/Option/Option.h"
20 #include "llvm/Support/BinaryStreamError.h"
21 #include "llvm/Support/Error.h"
22 #include "llvm/Support/InitLLVM.h"
23 #include "llvm/Support/ManagedStatic.h"
24 #include "llvm/Support/Path.h"
25 #include "llvm/Support/PrettyStackTrace.h"
26 #include "llvm/Support/Process.h"
27 #include "llvm/Support/Signals.h"
28 #include "llvm/Support/raw_ostream.h"
30 #include <system_error>
33 using namespace object
;
38 OPT_INVALID
= 0, // This is not an option ID.
39 #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
40 HELPTEXT, METAVAR, VALUES) \
46 #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
50 static const opt::OptTable::Info InfoTable
[] = {
51 #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
52 HELPTEXT, METAVAR, VALUES) \
54 PREFIX, NAME, HELPTEXT, \
55 METAVAR, OPT_##ID, opt::Option::KIND##Class, \
56 PARAM, FLAGS, OPT_##GROUP, \
57 OPT_##ALIAS, ALIASARGS, VALUES},
62 class CvtResOptTable
: public opt::OptTable
{
64 CvtResOptTable() : OptTable(InfoTable
, true) {}
68 LLVM_ATTRIBUTE_NORETURN
void reportError(Twine Msg
) {
73 static void reportError(StringRef Input
, std::error_code EC
) {
74 reportError(Twine(Input
) + ": " + EC
.message() + ".\n");
77 void error(std::error_code EC
) {
80 reportError(EC
.message() + ".\n");
83 void error(Error EC
) {
86 handleAllErrors(std::move(EC
),
87 [&](const ErrorInfoBase
&EI
) { reportError(EI
.message()); });
90 template <typename T
> T
error(Expected
<T
> EC
) {
92 error(EC
.takeError());
93 return std::move(EC
.get());
96 int main(int Argc
, const char **Argv
) {
97 InitLLVM
X(Argc
, Argv
);
101 ArrayRef
<const char *> ArgsArr
= makeArrayRef(Argv
+ 1, Argc
- 1);
102 opt::InputArgList InputArgs
= T
.ParseArgs(ArgsArr
, MAI
, MAC
);
104 if (InputArgs
.hasArg(OPT_HELP
)) {
105 T
.PrintHelp(outs(), "llvm-cvtres [options] file...", "Resource Converter", false);
109 bool Verbose
= InputArgs
.hasArg(OPT_VERBOSE
);
111 COFF::MachineTypes MachineType
;
113 if (InputArgs
.hasArg(OPT_MACHINE
)) {
114 std::string MachineString
= InputArgs
.getLastArgValue(OPT_MACHINE
).upper();
115 MachineType
= StringSwitch
<COFF::MachineTypes
>(MachineString
)
116 .Case("ARM", COFF::IMAGE_FILE_MACHINE_ARMNT
)
117 .Case("ARM64", COFF::IMAGE_FILE_MACHINE_ARM64
)
118 .Case("X64", COFF::IMAGE_FILE_MACHINE_AMD64
)
119 .Case("X86", COFF::IMAGE_FILE_MACHINE_I386
)
120 .Default(COFF::IMAGE_FILE_MACHINE_UNKNOWN
);
121 if (MachineType
== COFF::IMAGE_FILE_MACHINE_UNKNOWN
)
122 reportError("Unsupported machine architecture");
125 outs() << "Machine architecture not specified; assumed X64.\n";
126 MachineType
= COFF::IMAGE_FILE_MACHINE_AMD64
;
129 std::vector
<std::string
> InputFiles
= InputArgs
.getAllArgValues(OPT_INPUT
);
131 if (InputFiles
.size() == 0) {
132 reportError("No input file specified.\n");
135 SmallString
<128> OutputFile
;
137 if (InputArgs
.hasArg(OPT_OUT
)) {
138 OutputFile
= InputArgs
.getLastArgValue(OPT_OUT
);
140 OutputFile
= sys::path::filename(StringRef(InputFiles
[0]));
141 sys::path::replace_extension(OutputFile
, ".obj");
145 outs() << "Machine: ";
146 switch (MachineType
) {
147 case COFF::IMAGE_FILE_MACHINE_ARM64
:
150 case COFF::IMAGE_FILE_MACHINE_ARMNT
:
153 case COFF::IMAGE_FILE_MACHINE_I386
:
161 WindowsResourceParser Parser
;
163 for (const auto &File
: InputFiles
) {
164 Expected
<OwningBinary
<Binary
>> BinaryOrErr
= createBinary(File
);
166 reportError(File
, errorToErrorCode(BinaryOrErr
.takeError()));
168 Binary
&Binary
= *BinaryOrErr
.get().getBinary();
170 WindowsResource
*RF
= dyn_cast
<WindowsResource
>(&Binary
);
172 reportError(File
+ ": unrecognized file format.\n");
176 ResourceEntryRef Entry
= error(RF
->getHeadEntry());
179 error(Entry
.moveNext(End
));
182 outs() << "Number of resources: " << EntryNumber
<< "\n";
185 error(Parser
.parse(RF
));
189 Parser
.printTree(outs());
192 std::unique_ptr
<MemoryBuffer
> OutputBuffer
=
193 error(llvm::object::writeWindowsResourceCOFF(MachineType
, Parser
));
195 FileOutputBuffer::create(OutputFile
, OutputBuffer
->getBufferSize());
197 reportError(OutputFile
, errorToErrorCode(FileOrErr
.takeError()));
198 std::unique_ptr
<FileOutputBuffer
> FileBuffer
= std::move(*FileOrErr
);
199 std::copy(OutputBuffer
->getBufferStart(), OutputBuffer
->getBufferEnd(),
200 FileBuffer
->getBufferStart());
201 error(FileBuffer
->commit());
204 Expected
<OwningBinary
<Binary
>> BinaryOrErr
= createBinary(OutputFile
);
206 reportError(OutputFile
, errorToErrorCode(BinaryOrErr
.takeError()));
207 Binary
&Binary
= *BinaryOrErr
.get().getBinary();
208 ScopedPrinter
W(errs());
209 W
.printBinaryBlock("Output File Raw Data", Binary
.getData());