int main(int Argc, const char **Argv) { InitLLVM X(Argc, Argv); ExitOnErr.setBanner("llvm-rc: "); RcOptTable T; unsigned MAI, MAC; const char **DashDash = std::find_if( Argv + 1, Argv + Argc, [](StringRef Str) { return Str == "--"; }); ArrayRef<const char *> ArgsArr = makeArrayRef(Argv + 1, DashDash); opt::InputArgList InputArgs = T.ParseArgs(ArgsArr, MAI, MAC); // The tool prints nothing when invoked with no command-line arguments. if (InputArgs.hasArg(OPT_HELP)) { T.PrintHelp(outs(), "rc [options] file...", "Resource Converter", false); return 0; } const bool BeVerbose = InputArgs.hasArg(OPT_VERBOSE); std::vector<std::string> InArgsInfo = InputArgs.getAllArgValues(OPT_INPUT); if (DashDash != Argv + Argc) InArgsInfo.insert(InArgsInfo.end(), DashDash + 1, Argv + Argc); if (InArgsInfo.size() != 1) { fatalError("Exactly one input file should be provided."); } // Read and tokenize the input file. ErrorOr<std::unique_ptr<MemoryBuffer>> File = MemoryBuffer::getFile(InArgsInfo[0]); if (!File) { fatalError("Error opening file '" + Twine(InArgsInfo[0]) + "': " + File.getError().message()); } std::unique_ptr<MemoryBuffer> FileContents = std::move(*File); StringRef Contents = FileContents->getBuffer(); std::string FilteredContents = filterCppOutput(Contents); std::vector<RCToken> Tokens = ExitOnErr(tokenizeRC(FilteredContents)); if (BeVerbose) { const Twine TokenNames[] = { #define TOKEN(Name) #Name, #define SHORT_TOKEN(Name, Ch) #Name, #include "ResourceScriptTokenList.def" }; for (const RCToken &Token : Tokens) { outs() << TokenNames[static_cast<int>(Token.kind())] << ": " << Token.value(); if (Token.kind() == RCToken::Kind::Int) outs() << "; int value = " << Token.intValue(); outs() << "\n"; } } WriterParams Params; SmallString<128> InputFile(InArgsInfo[0]); llvm::sys::fs::make_absolute(InputFile); Params.InputFilePath = InputFile; Params.Include = InputArgs.getAllArgValues(OPT_INCLUDE); Params.NoInclude = InputArgs.getAllArgValues(OPT_NOINCLUDE); if (InputArgs.hasArg(OPT_CODEPAGE)) { if (InputArgs.getLastArgValue(OPT_CODEPAGE) .getAsInteger(10, Params.CodePage)) fatalError("Invalid code page: " + InputArgs.getLastArgValue(OPT_CODEPAGE)); switch (Params.CodePage) { case CpAcp: case CpWin1252: case CpUtf8: break; default: fatalError( "Unsupported code page, only 0, 1252 and 65001 are supported!"); } } std::unique_ptr<ResourceFileWriter> Visitor; bool IsDryRun = InputArgs.hasArg(OPT_DRY_RUN); if (!IsDryRun) { auto OutArgsInfo = InputArgs.getAllArgValues(OPT_FILEOUT); if (OutArgsInfo.empty()) { SmallString<128> OutputFile = InputFile; llvm::sys::path::replace_extension(OutputFile, "res"); OutArgsInfo.push_back(OutputFile.str()); } if (OutArgsInfo.size() != 1) fatalError( "No more than one output file should be provided (using /FO flag)."); std::error_code EC; auto FOut = llvm::make_unique<raw_fd_ostream>( OutArgsInfo[0], EC, sys::fs::FA_Read | sys::fs::FA_Write); if (EC) fatalError("Error opening output file '" + OutArgsInfo[0] + "': " + EC.message()); Visitor = llvm::make_unique<ResourceFileWriter>(Params, std::move(FOut)); Visitor->AppendNull = InputArgs.hasArg(OPT_ADD_NULL); ExitOnErr(NullResource().visit(Visitor.get())); // Set the default language; choose en-US arbitrarily. ExitOnErr(LanguageResource(0x09, 0x01).visit(Visitor.get())); } rc::RCParser Parser{std::move(Tokens)}; while (!Parser.isEof()) { auto Resource = ExitOnErr(Parser.parseSingleResource()); if (BeVerbose) Resource->log(outs()); if (!IsDryRun) ExitOnErr(Resource->visit(Visitor.get())); } // STRINGTABLE resources come at the very end. if (!IsDryRun) ExitOnErr(Visitor->dumpAllStringTables()); return 0; }
int main(int argc_, const char *argv_[]) { sys::PrintStackTraceOnErrorSignal(argv_[0]); PrettyStackTraceProgram X(argc_, argv_); ExitOnErr.setBanner("llvm-rc: "); SmallVector<const char *, 256> argv; SpecificBumpPtrAllocator<char> ArgAllocator; ExitOnErr(errorCodeToError(sys::Process::GetArgumentVector( argv, makeArrayRef(argv_, argc_), ArgAllocator))); llvm_shutdown_obj Y; RcOptTable T; unsigned MAI, MAC; ArrayRef<const char *> ArgsArr = makeArrayRef(argv_ + 1, argc_); opt::InputArgList InputArgs = T.ParseArgs(ArgsArr, MAI, MAC); // The tool prints nothing when invoked with no command-line arguments. if (InputArgs.hasArg(OPT_HELP)) { T.PrintHelp(outs(), "rc", "Resource Converter", false); return 0; } const bool BeVerbose = InputArgs.hasArg(OPT_VERBOSE); std::vector<std::string> InArgsInfo = InputArgs.getAllArgValues(OPT_INPUT); if (InArgsInfo.size() != 1) { fatalError("Exactly one input file should be provided."); } // Read and tokenize the input file. const Twine &Filename = InArgsInfo[0]; ErrorOr<std::unique_ptr<MemoryBuffer>> File = MemoryBuffer::getFile(Filename); if (!File) { fatalError("Error opening file '" + Filename + "': " + File.getError().message()); } std::unique_ptr<MemoryBuffer> FileContents = std::move(*File); StringRef Contents = FileContents->getBuffer(); std::vector<RCToken> Tokens = ExitOnErr(tokenizeRC(Contents)); if (BeVerbose) { const Twine TokenNames[] = { #define TOKEN(Name) #Name, #define SHORT_TOKEN(Name, Ch) #Name, #include "ResourceScriptTokenList.h" #undef TOKEN #undef SHORT_TOKEN }; for (const RCToken &Token : Tokens) { outs() << TokenNames[static_cast<int>(Token.kind())] << ": " << Token.value(); if (Token.kind() == RCToken::Kind::Int) outs() << "; int value = " << Token.intValue(); outs() << "\n"; } } std::unique_ptr<ResourceFileWriter> Visitor; bool IsDryRun = InputArgs.hasArg(OPT_DRY_RUN); if (!IsDryRun) { auto OutArgsInfo = InputArgs.getAllArgValues(OPT_FILEOUT); if (OutArgsInfo.size() != 1) fatalError( "Exactly one output file should be provided (using /FO flag)."); std::error_code EC; auto FOut = llvm::make_unique<raw_fd_ostream>(OutArgsInfo[0], EC, sys::fs::F_RW); if (EC) fatalError("Error opening output file '" + OutArgsInfo[0] + "': " + EC.message()); Visitor = llvm::make_unique<ResourceFileWriter>(std::move(FOut)); Visitor->AppendNull = InputArgs.hasArg(OPT_ADD_NULL); ExitOnErr(NullResource().visit(Visitor.get())); // Set the default language; choose en-US arbitrarily. ExitOnErr(LanguageResource(0x09, 0x01).visit(Visitor.get())); } rc::RCParser Parser{std::move(Tokens)}; while (!Parser.isEof()) { auto Resource = ExitOnErr(Parser.parseSingleResource()); if (BeVerbose) Resource->log(outs()); if (!IsDryRun) ExitOnErr(Resource->visit(Visitor.get())); } // STRINGTABLE resources come at the very end. if (!IsDryRun) ExitOnErr(Visitor->dumpAllStringTables()); return 0; }