123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234 |
- //===- llvm-cvtres.cpp - Serialize .res files into .obj ---------*- 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
- //
- //===----------------------------------------------------------------------===//
- //
- // Serialize .res files into .obj files. This is intended to be a
- // platform-independent port of Microsoft's cvtres.exe.
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/BinaryFormat/Magic.h"
- #include "llvm/Object/Binary.h"
- #include "llvm/Object/WindowsMachineFlag.h"
- #include "llvm/Object/WindowsResource.h"
- #include "llvm/Option/Arg.h"
- #include "llvm/Option/ArgList.h"
- #include "llvm/Option/Option.h"
- #include "llvm/Support/BinaryStreamError.h"
- #include "llvm/Support/Error.h"
- #include "llvm/Support/InitLLVM.h"
- #include "llvm/Support/ManagedStatic.h"
- #include "llvm/Support/Path.h"
- #include "llvm/Support/PrettyStackTrace.h"
- #include "llvm/Support/Process.h"
- #include "llvm/Support/ScopedPrinter.h"
- #include "llvm/Support/Signals.h"
- #include "llvm/Support/raw_ostream.h"
- #include <system_error>
- using namespace llvm;
- using namespace object;
- 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
- 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) {}
- };
- }
- [[noreturn]] static void reportError(Twine Msg) {
- errs() << Msg;
- exit(1);
- }
- static void reportError(StringRef Input, std::error_code EC) {
- reportError(Twine(Input) + ": " + EC.message() + ".\n");
- }
- static void error(StringRef Input, Error EC) {
- if (!EC)
- return;
- handleAllErrors(std::move(EC), [&](const ErrorInfoBase &EI) {
- reportError(Twine(Input) + ": " + EI.message() + ".\n");
- });
- }
- static void error(Error EC) {
- if (!EC)
- return;
- handleAllErrors(std::move(EC),
- [&](const ErrorInfoBase &EI) { reportError(EI.message()); });
- }
- static uint32_t getTime() {
- std::time_t Now = time(nullptr);
- if (Now < 0 || !isUInt<32>(Now))
- return UINT32_MAX;
- return static_cast<uint32_t>(Now);
- }
- template <typename T> T error(Expected<T> EC) {
- if (!EC)
- error(EC.takeError());
- return std::move(EC.get());
- }
- template <typename T> T error(StringRef Input, Expected<T> EC) {
- if (!EC)
- error(Input, EC.takeError());
- return std::move(EC.get());
- }
- template <typename T> T error(StringRef Input, ErrorOr<T> &&EC) {
- return error(Input, errorOrToExpected(std::move(EC)));
- }
- 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);
- if (InputArgs.hasArg(OPT_HELP)) {
- T.printHelp(outs(), "llvm-cvtres [options] file...", "Resource Converter");
- return 0;
- }
- bool Verbose = InputArgs.hasArg(OPT_VERBOSE);
- COFF::MachineTypes MachineType;
- if (opt::Arg *Arg = InputArgs.getLastArg(OPT_MACHINE)) {
- MachineType = getMachineType(Arg->getValue());
- if (MachineType == COFF::IMAGE_FILE_MACHINE_UNKNOWN) {
- reportError(Twine("Unsupported machine architecture ") + Arg->getValue() +
- "\n");
- }
- } else {
- if (Verbose)
- outs() << "Machine architecture not specified; assumed X64.\n";
- MachineType = COFF::IMAGE_FILE_MACHINE_AMD64;
- }
- std::vector<std::string> InputFiles = InputArgs.getAllArgValues(OPT_INPUT);
- if (InputFiles.size() == 0) {
- reportError("No input file specified.\n");
- }
- SmallString<128> OutputFile;
- if (opt::Arg *Arg = InputArgs.getLastArg(OPT_OUT)) {
- OutputFile = Arg->getValue();
- } else {
- OutputFile = sys::path::filename(StringRef(InputFiles[0]));
- sys::path::replace_extension(OutputFile, ".obj");
- }
- uint32_t DateTimeStamp;
- if (llvm::opt::Arg *Arg = InputArgs.getLastArg(OPT_TIMESTAMP)) {
- StringRef Value(Arg->getValue());
- if (Value.getAsInteger(0, DateTimeStamp))
- reportError(Twine("invalid timestamp: ") + Value +
- ". Expected 32-bit integer\n");
- } else {
- DateTimeStamp = getTime();
- }
- if (Verbose)
- outs() << "Machine: " << machineToStr(MachineType) << '\n';
- WindowsResourceParser Parser;
- for (const auto &File : InputFiles) {
- std::unique_ptr<MemoryBuffer> Buffer = error(
- File, MemoryBuffer::getFileOrSTDIN(File, /*IsText=*/false,
- /*RequiresNullTerminator=*/false));
- file_magic Type = identify_magic(Buffer->getMemBufferRef().getBuffer());
- if (Type != file_magic::windows_resource)
- reportError(File + ": unrecognized file format.\n");
- std::unique_ptr<WindowsResource> Binary = error(
- File,
- WindowsResource::createWindowsResource(Buffer->getMemBufferRef()));
- WindowsResource *RF = Binary.get();
- if (Verbose) {
- int EntryNumber = 0;
- ResourceEntryRef Entry = error(RF->getHeadEntry());
- bool End = false;
- while (!End) {
- error(Entry.moveNext(End));
- EntryNumber++;
- }
- outs() << "Number of resources: " << EntryNumber << "\n";
- }
- std::vector<std::string> Duplicates;
- error(Parser.parse(RF, Duplicates));
- for (const auto& DupeDiag : Duplicates)
- reportError(DupeDiag);
- }
- if (Verbose) {
- Parser.printTree(outs());
- }
- std::unique_ptr<MemoryBuffer> OutputBuffer =
- error(llvm::object::writeWindowsResourceCOFF(MachineType, Parser,
- DateTimeStamp));
- auto 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());
- if (Verbose) {
- std::unique_ptr<MemoryBuffer> Buffer =
- error(OutputFile,
- MemoryBuffer::getFileOrSTDIN(OutputFile, /*IsText=*/false,
- /*RequiresNullTerminator=*/false));
- ScopedPrinter W(errs());
- W.printBinaryBlock("Output File Raw Data",
- Buffer->getMemBufferRef().getBuffer());
- }
- return 0;
- }
|