|  | //===- lib/Driver/CoreDriver.cpp ------------------------------------------===// | 
|  | // | 
|  | //                             The LLVM Linker | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "lld/Core/Reader.h" | 
|  | #include "lld/Driver/Driver.h" | 
|  | #include "lld/ReaderWriter/CoreLinkingContext.h" | 
|  | #include "llvm/ADT/ArrayRef.h" | 
|  | #include "llvm/ADT/STLExtras.h" | 
|  | #include "llvm/ADT/Triple.h" | 
|  | #include "llvm/Option/Arg.h" | 
|  | #include "llvm/Option/Option.h" | 
|  | #include "llvm/Support/CommandLine.h" | 
|  | #include "llvm/Support/FileSystem.h" | 
|  | #include "llvm/Support/Host.h" | 
|  | #include "llvm/Support/ManagedStatic.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Support/PrettyStackTrace.h" | 
|  | #include "llvm/Support/Signals.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  |  | 
|  | using namespace lld; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Create enum with OPT_xxx values for each option in CoreOptions.td | 
|  | enum { | 
|  | OPT_INVALID = 0, | 
|  | #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ | 
|  | HELP, META) \ | 
|  | OPT_##ID, | 
|  | #include "CoreOptions.inc" | 
|  | #undef OPTION | 
|  | }; | 
|  |  | 
|  | // Create prefix string literals used in CoreOptions.td | 
|  | #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE; | 
|  | #include "CoreOptions.inc" | 
|  | #undef PREFIX | 
|  |  | 
|  | // Create table mapping all options defined in CoreOptions.td | 
|  | static const llvm::opt::OptTable::Info infoTable[] = { | 
|  | #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ | 
|  | HELPTEXT, METAVAR)   \ | 
|  | { PREFIX, NAME, HELPTEXT, METAVAR, OPT_##ID, llvm::opt::Option::KIND##Class, \ | 
|  | PARAM, FLAGS, OPT_##GROUP, OPT_##ALIAS, ALIASARGS }, | 
|  | #include "CoreOptions.inc" | 
|  | #undef OPTION | 
|  | }; | 
|  |  | 
|  | // Create OptTable class for parsing actual command line arguments | 
|  | class CoreOptTable : public llvm::opt::OptTable { | 
|  | public: | 
|  | CoreOptTable() : OptTable(infoTable, llvm::array_lengthof(infoTable)){} | 
|  | }; | 
|  |  | 
|  | } // namespace anonymous | 
|  |  | 
|  |  | 
|  | namespace lld { | 
|  |  | 
|  | static const Registry::KindStrings coreKindStrings[] = { | 
|  | { CoreLinkingContext::TEST_RELOC_CALL32,        "call32" }, | 
|  | { CoreLinkingContext::TEST_RELOC_PCREL32,       "pcrel32" }, | 
|  | { CoreLinkingContext::TEST_RELOC_GOT_LOAD32,    "gotLoad32" }, | 
|  | { CoreLinkingContext::TEST_RELOC_GOT_USE32,     "gotUse32" }, | 
|  | { CoreLinkingContext::TEST_RELOC_LEA32_WAS_GOT, "lea32wasGot" }, | 
|  | LLD_KIND_STRING_END | 
|  | }; | 
|  |  | 
|  | bool CoreDriver::link(int argc, const char *argv[], raw_ostream &diagnostics) { | 
|  | CoreLinkingContext ctx; | 
|  |  | 
|  | // Register possible input file parsers. | 
|  | ctx.registry().addSupportNativeObjects(); | 
|  | ctx.registry().addSupportYamlFiles(); | 
|  | ctx.registry().addKindTable(Reference::KindNamespace::testing, | 
|  | Reference::KindArch::all, coreKindStrings); | 
|  |  | 
|  | if (!parse(argc, argv, ctx)) | 
|  | return false; | 
|  | return Driver::link(ctx); | 
|  | } | 
|  |  | 
|  | bool CoreDriver::parse(int argc, const char *argv[], CoreLinkingContext &ctx, | 
|  | raw_ostream &diagnostics) { | 
|  | // Parse command line options using CoreOptions.td | 
|  | std::unique_ptr<llvm::opt::InputArgList> parsedArgs; | 
|  | CoreOptTable table; | 
|  | unsigned missingIndex; | 
|  | unsigned missingCount; | 
|  | parsedArgs.reset( | 
|  | table.ParseArgs(&argv[1], &argv[argc], missingIndex, missingCount)); | 
|  | if (missingCount) { | 
|  | diagnostics << "error: missing arg value for '" | 
|  | << parsedArgs->getArgString(missingIndex) << "' expected " | 
|  | << missingCount << " argument(s).\n"; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Set default options | 
|  | ctx.setOutputPath("-"); | 
|  | ctx.setDeadStripping(false); | 
|  | ctx.setGlobalsAreDeadStripRoots(false); | 
|  | ctx.setPrintRemainingUndefines(false); | 
|  | ctx.setAllowRemainingUndefines(true); | 
|  | ctx.setSearchArchivesToOverrideTentativeDefinitions(false); | 
|  |  | 
|  | // Process all the arguments and create input files. | 
|  | for (auto inputArg : *parsedArgs) { | 
|  | switch (inputArg->getOption().getID()) { | 
|  | case OPT_mllvm: | 
|  | ctx.appendLLVMOption(inputArg->getValue()); | 
|  | break; | 
|  |  | 
|  | case OPT_entry: | 
|  | ctx.setEntrySymbolName(inputArg->getValue()); | 
|  | break; | 
|  |  | 
|  | case OPT_output: | 
|  | ctx.setOutputPath(inputArg->getValue()); | 
|  | break; | 
|  |  | 
|  | case OPT_dead_strip: | 
|  | ctx.setDeadStripping(true); | 
|  | break; | 
|  |  | 
|  | case OPT_keep_globals: | 
|  | ctx.setGlobalsAreDeadStripRoots(true); | 
|  | break; | 
|  |  | 
|  | case OPT_undefines_are_errors: | 
|  | ctx.setPrintRemainingUndefines(true); | 
|  | ctx.setAllowRemainingUndefines(false); | 
|  | break; | 
|  |  | 
|  | case OPT_commons_search_archives: | 
|  | ctx.setSearchArchivesToOverrideTentativeDefinitions(true); | 
|  | break; | 
|  |  | 
|  | case OPT_add_pass: | 
|  | ctx.addPassNamed(inputArg->getValue()); | 
|  | break; | 
|  |  | 
|  | case OPT_INPUT: { | 
|  | std::vector<std::unique_ptr<File>> files | 
|  | = loadFile(ctx, inputArg->getValue(), false); | 
|  | for (std::unique_ptr<File> &file : files) | 
|  | ctx.getNodes().push_back(llvm::make_unique<FileNode>(std::move(file))); | 
|  | break; | 
|  | } | 
|  |  | 
|  | default: | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ctx.getNodes().empty()) { | 
|  | diagnostics << "No input files\n"; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Validate the combination of options used. | 
|  | return ctx.validate(diagnostics); | 
|  | } | 
|  |  | 
|  | } // namespace lld |