//===- Main.cpp -----------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include <mcld/Environment.h> #include <mcld/IRBuilder.h> #include <mcld/Linker.h> #include <mcld/LinkerConfig.h> #include <mcld/LinkerScript.h> #include <mcld/Module.h> #include <mcld/ADT/StringEntry.h> #include <mcld/MC/InputAction.h> #include <mcld/MC/CommandAction.h> #include <mcld/MC/FileAction.h> #include <mcld/MC/ZOption.h> #include <mcld/Support/raw_ostream.h> #include <mcld/Support/MsgHandling.h> #include <mcld/Support/Path.h> #include <mcld/Support/SystemUtils.h> #include <mcld/Support/TargetRegistry.h> #include <llvm/ADT/ArrayRef.h> #include <llvm/ADT/SmallVector.h> #include <llvm/ADT/STLExtras.h> #include <llvm/ADT/StringRef.h> #include <llvm/ADT/StringSwitch.h> #include <llvm/Option/Arg.h> #include <llvm/Option/ArgList.h> #include <llvm/Option/OptTable.h> #include <llvm/Option/Option.h> #include <llvm/Support/ManagedStatic.h> #include <llvm/Support/Process.h> #include <llvm/Support/Signals.h> #include <cassert> #include <cstdlib> #include <string> #if defined(HAVE_UNISTD_H) #include <unistd.h> #endif #if defined(_MSC_VER) || defined(__MINGW32__) #include <io.h> #ifndef STDIN_FILENO #define STDIN_FILENO 0 #endif #ifndef STDOUT_FILENO #define STDOUT_FILENO 1 #endif #ifndef STDERR_FILENO #define STDERR_FILENO 2 #endif #endif namespace { class Driver { private: enum Option { // This is not an option. kOpt_INVALID = 0, #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ HELPTEXT, METAVAR) \ kOpt_ ## ID, #include "Options.inc" // NOLINT #undef OPTION kOpt_LastOption }; class OptTable : public llvm::opt::OptTable { private: #define PREFIX(NAME, VALUE) \ static const char* const NAME[]; #include "Options.inc" // NOLINT #undef PREFIX static const llvm::opt::OptTable::Info InfoTable[]; public: OptTable(); }; private: explicit Driver(const char* prog_name) : prog_name_(prog_name), module_(script_), ir_builder_(module_, config_) { return; } public: static std::unique_ptr<Driver> Create(llvm::ArrayRef<const char*> argv); bool Run(); private: bool TranslateArguments(llvm::opt::InputArgList& args); private: const char* prog_name_; mcld::LinkerScript script_; mcld::LinkerConfig config_; mcld::Module module_; mcld::IRBuilder ir_builder_; mcld::Linker linker_; private: DISALLOW_COPY_AND_ASSIGN(Driver); }; #define PREFIX(NAME, VALUE) \ const char* const Driver::OptTable::NAME[] = VALUE; #include "Options.inc" // NOLINT #undef PREFIX const llvm::opt::OptTable::Info Driver::OptTable::InfoTable[] = { #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \ HELPTEXT, METAVAR) \ { PREFIX, NAME, HELPTEXT, METAVAR, kOpt_ ## ID, \ llvm::opt::Option::KIND ## Class, PARAM, FLAGS, kOpt_ ## GROUP, \ kOpt_ ## ALIAS, ALIASARGS }, #include "Options.inc" // NOLINT #undef OPTION }; Driver::OptTable::OptTable() : llvm::opt::OptTable(InfoTable) { } inline bool ShouldColorize() { const char* term = getenv("TERM"); return term && (0 != strcmp(term, "dumb")); } /// ParseProgName - Parse program name /// This function simplifies cross-compiling by reading triple from the program /// name. For example, if the program name is `arm-linux-eabi-ld.mcld', we can /// get the triple is arm-linux-eabi by the program name. inline std::string ParseProgName(const char* prog_name) { static const char* suffixes[] = {"ld", "ld.mcld"}; std::string name(mcld::sys::fs::Path(prog_name).stem().native()); for (size_t i = 0; i < sizeof(suffixes) / sizeof(suffixes[0]); ++i) { if (name == suffixes[i]) return std::string(); } llvm::StringRef prog_name_ref(prog_name); llvm::StringRef prefix; for (size_t i = 0; i < sizeof(suffixes) / sizeof(suffixes[0]); ++i) { if (!prog_name_ref.endswith(suffixes[i])) continue; llvm::StringRef::size_type last_component = prog_name_ref.rfind('-', prog_name_ref.size() - strlen(suffixes[i])); if (last_component == llvm::StringRef::npos) continue; llvm::StringRef prefix = prog_name_ref.slice(0, last_component); std::string ignored_error; if (!mcld::TargetRegistry::lookupTarget(prefix, ignored_error)) continue; return prefix.str(); } return std::string(); } inline void ParseEmulation(llvm::Triple& triple, const char* emulation) { llvm::Triple emu_triple = llvm::StringSwitch<llvm::Triple>(emulation) .Case("aarch64linux", llvm::Triple("aarch64", "", "linux", "gnu")) .Case("armelf_linux_eabi", llvm::Triple("arm", "", "linux", "gnu")) .Case("elf_i386", llvm::Triple("i386", "", "", "gnu")) .Case("elf_x86_64", llvm::Triple("x86_64", "", "", "gnu")) .Case("elf32_x86_64", llvm::Triple("x86_64", "", "", "gnux32")) .Case("elf_i386_fbsd", llvm::Triple("i386", "", "freebsd", "gnu")) .Case("elf_x86_64_fbsd", llvm::Triple("x86_64", "", "freebsd", "gnu")) .Case("elf32ltsmip", llvm::Triple("mipsel", "", "", "gnu")) .Case("elf64ltsmip", llvm::Triple("mips64el", "", "", "gnu")) .Default(llvm::Triple()); if (emu_triple.getArch() == llvm::Triple::UnknownArch && emu_triple.getOS() == llvm::Triple::UnknownOS && emu_triple.getEnvironment() == llvm::Triple::UnknownEnvironment) mcld::error(mcld::diag::err_invalid_emulation) << emulation << "\n"; if (emu_triple.getArch() != llvm::Triple::UnknownArch) triple.setArch(emu_triple.getArch()); if (emu_triple.getOS() != llvm::Triple::UnknownOS) triple.setOS(emu_triple.getOS()); if (emu_triple.getEnvironment() != llvm::Triple::UnknownEnvironment) triple.setEnvironment(emu_triple.getEnvironment()); } /// Configure the output filename. inline bool ConfigureOutputName(llvm::StringRef output_name, mcld::Module& module, mcld::LinkerConfig& config) { std::string output(output_name.str()); if (output.empty()) { if (config.targets().triple().getOS() == llvm::Triple::Win32) { output.assign("_out"); switch (config.codeGenType()) { case mcld::LinkerConfig::Object: { output += ".obj"; break; } case mcld::LinkerConfig::DynObj: { output += ".dll"; break; } case mcld::LinkerConfig::Exec: { output += ".exe"; break; } case mcld::LinkerConfig::External: break; default: { return false; break; } } // switch (config.codeGenType()) } else { output.assign("a.out"); } } // if (output.empty()) module.setName(output); return true; } bool InitializeInputs(mcld::IRBuilder& ir_builder, std::vector<std::unique_ptr<mcld::InputAction>>& input_actions) { for (auto& action : input_actions) { assert(action != nullptr); action->activate(ir_builder.getInputBuilder()); } if (ir_builder.getInputBuilder().isInGroup()) { mcld::fatal(mcld::diag::fatal_forbid_nest_group); return false; } return true; } bool Driver::TranslateArguments(llvm::opt::InputArgList& args) { //===--------------------------------------------------------------------===// // Preference //===--------------------------------------------------------------------===// // --color=mode if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Color)) { bool res = llvm::StringSwitch<bool>(arg->getValue()) .Case("never", false) .Case("always", true) .Case("auto", ShouldColorize() && llvm::sys::Process::FileDescriptorIsDisplayed( STDOUT_FILENO)) .Default(false); config_.options().setColor(res); mcld::outs().setColor(res); mcld::errs().setColor(res); } // --trace config_.options().setTrace(args.hasArg(kOpt_Trace)); // --verbose=level if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Verbose)) { llvm::StringRef value = arg->getValue(); int level; if (value.getAsInteger(0, level)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue(); return false; } config_.options().setVerbose(level); } // --error-limit NUMBER if (llvm::opt::Arg* arg = args.getLastArg(kOpt_ErrorLimit)) { llvm::StringRef value = arg->getValue(); int num; if (value.getAsInteger(0, num) || (num < 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue(); return false; } config_.options().setMaxErrorNum(num); } // --warning-limit NUMBER if (llvm::opt::Arg* arg = args.getLastArg(kOpt_WarningLimit)) { llvm::StringRef value = arg->getValue(); int num; if (value.getAsInteger(0, num) || (num < 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue(); return false; } config_.options().setMaxWarnNum(num); } // --warn-shared-textrel config_.options().setWarnSharedTextrel(args.hasArg(kOpt_WarnSharedTextrel)); //===--------------------------------------------------------------------===// // Target //===--------------------------------------------------------------------===// llvm::Triple triple; if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Triple)) { // 1. Use the triple from command. // -mtriple=value triple.setTriple(arg->getValue()); } else { std::string prog_triple = ParseProgName(prog_name_); if (!prog_triple.empty()) { // 2. Use the triple from the program name prefix. triple.setTriple(prog_triple); } else { // 3. Use the default target triple. triple.setTriple(mcld::sys::getDefaultTargetTriple()); } } // If a specific emulation was requested, apply it now. if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Emulation)) { // -m emulation ParseEmulation(triple, arg->getValue()); } else if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Arch)) { // -march=value config_.targets().setArch(arg->getValue()); } if (llvm::opt::Arg* arg = args.getLastArg(kOpt_CPU)) { config_.targets().setTargetCPU(arg->getValue()); } config_.targets().setTriple(triple); // --gpsize=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_GPSize)) { llvm::StringRef value = arg->getValue(); int size; if (value.getAsInteger(0, size) || (size< 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } config_.targets().setGPSize(size); } // --stub-group-size=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_StubGroupSize)) { llvm::StringRef value = arg->getValue(); int size; if (value.getAsInteger(0, size) || (size< 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } config_.targets().setStubGroupSize(size); } // --fix-cortex-a53-835769 config_.targets().setFixCA53Erratum835769( args.hasArg(kOpt_FixCA53Erratum835769)); // --fix-cortex-a53-843419 config_.targets().setFixCA53Erratum843419( args.hasArg(kOpt_FixCA53Erratum843419)); //===--------------------------------------------------------------------===// // Dynamic //===--------------------------------------------------------------------===// // --entry=entry if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Entry)) { script_.setEntry(arg->getValue()); } // -Bsymbolic config_.options().setBsymbolic(args.hasArg(kOpt_Bsymbolic)); // -Bgroup config_.options().setBgroup(args.hasArg(kOpt_Bgroup)); // -soname=name if (llvm::opt::Arg* arg = args.getLastArg(kOpt_SOName)) { config_.options().setSOName(arg->getValue()); } // --no-undefined if (args.hasArg(kOpt_NoUndef)) { config_.options().setNoUndefined(true); } // --allow-multiple-definition if (args.hasArg(kOpt_AllowMulDefs)) { config_.options().setMulDefs(true); } // -z options for (llvm::opt::Arg* arg : args.filtered(kOpt_Z)) { llvm::StringRef value = arg->getValue(); mcld::ZOption z_opt = llvm::StringSwitch<mcld::ZOption>(value) .Case("combreloc", mcld::ZOption(mcld::ZOption::CombReloc)) .Case("nocombreloc", mcld::ZOption(mcld::ZOption::NoCombReloc)) .Case("defs", mcld::ZOption(mcld::ZOption::Defs)) .Case("execstack", mcld::ZOption(mcld::ZOption::ExecStack)) .Case("noexecstack", mcld::ZOption(mcld::ZOption::NoExecStack)) .Case("initfirst", mcld::ZOption(mcld::ZOption::InitFirst)) .Case("interpose", mcld::ZOption(mcld::ZOption::InterPose)) .Case("loadfltr", mcld::ZOption(mcld::ZOption::LoadFltr)) .Case("muldefs", mcld::ZOption(mcld::ZOption::MulDefs)) .Case("nocopyreloc", mcld::ZOption(mcld::ZOption::NoCopyReloc)) .Case("nodefaultlib", mcld::ZOption(mcld::ZOption::NoDefaultLib)) .Case("nodelete", mcld::ZOption(mcld::ZOption::NoDelete)) .Case("nodlopen", mcld::ZOption(mcld::ZOption::NoDLOpen)) .Case("nodump", mcld::ZOption(mcld::ZOption::NoDump)) .Case("relro", mcld::ZOption(mcld::ZOption::Relro)) .Case("norelro", mcld::ZOption(mcld::ZOption::NoRelro)) .Case("lazy", mcld::ZOption(mcld::ZOption::Lazy)) .Case("now", mcld::ZOption(mcld::ZOption::Now)) .Case("origin", mcld::ZOption(mcld::ZOption::Origin)) .Default(mcld::ZOption()); if (z_opt.kind() == mcld::ZOption::Unknown) { if (value.startswith("common-page-size=")) { // -z common-page-size=value z_opt.setKind(mcld::ZOption::CommPageSize); long long unsigned size = 0; value.drop_front(17).getAsInteger(0, size); z_opt.setPageSize(static_cast<uint64_t>(size)); } else if (value.startswith("max-page-size=")) { // -z max-page-size=value z_opt.setKind(mcld::ZOption::MaxPageSize); long long unsigned size = 0; value.drop_front(14).getAsInteger(0, size); z_opt.setPageSize(static_cast<uint64_t>(size)); } } config_.options().addZOption(z_opt); } // --dynamic-linker=file if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Dyld)) { config_.options().setDyld(arg->getValue()); } // --enable-new-dtags config_.options().setNewDTags(args.hasArg(kOpt_EnableNewDTags)); // --spare-dyanmic-tags COUNT if (llvm::opt::Arg* arg = args.getLastArg(kOpt_SpareDTags)) { llvm::StringRef value = arg->getValue(); int num; if (value.getAsInteger(0, num) || (num < 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } config_.options().setNumSpareDTags(num); } //===--------------------------------------------------------------------===// // Output //===--------------------------------------------------------------------===// // Setup the codegen type. if (args.hasArg(kOpt_Shared) || args.hasArg(kOpt_PIE)) { // -shared, -pie config_.setCodeGenType(mcld::LinkerConfig::DynObj); } else if (args.hasArg(kOpt_Relocatable)) { // -r config_.setCodeGenType(mcld::LinkerConfig::Object); } else if (llvm::opt::Arg* arg = args.getLastArg(kOpt_OutputFormat)) { // --oformat=value llvm::StringRef value = arg->getValue(); if (value.equals("binary")) { config_.setCodeGenType(mcld::LinkerConfig::Binary); } } else { config_.setCodeGenType(mcld::LinkerConfig::Exec); } // Setup the output filename. llvm::StringRef output_name; if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Output)) { output_name = arg->getValue(); } if (!ConfigureOutputName(output_name, module_, config_)) { mcld::unreachable(mcld::diag::unrecognized_output_file) << module_.name(); return false; } else { if (!args.hasArg(kOpt_SOName)) { config_.options().setSOName(module_.name()); } } // --format=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_InputFormat)) { llvm::StringRef value = arg->getValue(); if (value.equals("binary")) { config_.options().setBinaryInput(); } } // Setup debug info stripping. config_.options().setStripDebug(args.hasArg(kOpt_StripDebug) || args.hasArg(kOpt_StripAll)); // Setup symbol stripping mode. if (args.hasArg(kOpt_StripAll)) { config_.options().setStripSymbols( mcld::GeneralOptions::StripSymbolMode::StripAllSymbols); } else if (args.hasArg(kOpt_DiscardAll)) { config_.options().setStripSymbols( mcld::GeneralOptions::StripSymbolMode::StripLocals); } else if (args.hasArg(kOpt_DiscardLocals)) { config_.options().setStripSymbols( mcld::GeneralOptions::StripSymbolMode::StripTemporaries); } else { config_.options().setStripSymbols( mcld::GeneralOptions::StripSymbolMode::KeepAllSymbols); } // --eh-frame-hdr config_.options().setEhFrameHdr(args.hasArg(kOpt_EHFrameHdr)); // -pie config_.options().setPIE(args.hasArg(kOpt_PIE)); // --nmagic config_.options().setNMagic(args.hasArg(kOpt_NMagic)); // --omagic config_.options().setOMagic(args.hasArg(kOpt_OMagic)); // --hash-style=style if (llvm::opt::Arg* arg = args.getLastArg(kOpt_HashStyle)) { mcld::GeneralOptions::HashStyle style = llvm::StringSwitch<mcld::GeneralOptions::HashStyle>(arg->getValue()) .Case("sysv", mcld::GeneralOptions::HashStyle::SystemV) .Case("gnu", mcld::GeneralOptions::HashStyle::GNU) .Case("both", mcld::GeneralOptions::HashStyle::Both) .Default(mcld::GeneralOptions::HashStyle::Unknown); if (style != mcld::GeneralOptions::HashStyle::Unknown) { config_.options().setHashStyle(style); } } // --[no]-export-dynamic if (llvm::opt::Arg* arg = args.getLastArg(kOpt_ExportDynamic, kOpt_NoExportDynamic)) { if (arg->getOption().matches(kOpt_ExportDynamic)) { config_.options().setExportDynamic(true); } else { config_.options().setExportDynamic(false); } } // --no-warn-mismatch config_.options().setWarnMismatch(!args.hasArg(kOpt_NoWarnMismatch)); // --exclude-libs if (llvm::opt::Arg* arg = args.getLastArg(kOpt_ExcludeLibs)) { llvm::StringRef value = arg->getValue(); do { std::pair<llvm::StringRef, llvm::StringRef> res = value.split(','); config_.options().excludeLIBS().insert(res.first.str()); value = res.second; } while (!value.empty()); } //===--------------------------------------------------------------------===// // Search Path //===--------------------------------------------------------------------===// // --sysroot if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Sysroot)) { mcld::sys::fs::Path path(arg->getValue()); if (mcld::sys::fs::exists(path) && mcld::sys::fs::is_directory(path)) { script_.setSysroot(path); } } // -L searchdir for (llvm::opt::Arg* arg : args.filtered(kOpt_LibraryPath)) { if (!script_.directories().insert(arg->getValue())) mcld::warning(mcld::diag::warn_cannot_open_search_dir) << arg->getValue(); } // -nostdlib config_.options().setNoStdlib(args.hasArg(kOpt_NoStdlib)); // -rpath=path for (llvm::opt::Arg* arg : args.filtered(kOpt_RPath)) { config_.options().getRpathList().push_back(arg->getValue()); } //===--------------------------------------------------------------------===// // Symbol //===--------------------------------------------------------------------===// // -d/-dc/-dp config_.options().setDefineCommon(args.hasArg(kOpt_DefineCommon)); // -u symbol for (llvm::opt::Arg* arg : args.filtered(kOpt_Undefined)) { config_.options().getUndefSymList().push_back(arg->getValue()); } //===--------------------------------------------------------------------===// // Script //===--------------------------------------------------------------------===// // --wrap=symbol for (llvm::opt::Arg* arg : args.filtered(kOpt_Wrap)) { bool exist = false; const char* symbol = arg->getValue(); // symbol -> __wrap_symbol mcld::StringEntry<llvm::StringRef>* to_wrap = script_.renameMap().insert(symbol, exist); std::string to_wrap_str; to_wrap_str.append("__wrap_") .append(symbol); to_wrap->setValue(to_wrap_str); if (exist) mcld::warning(mcld::diag::rewrap) << symbol << to_wrap_str; // __real_symbol -> symbol std::string from_real_str; to_wrap_str.append("__real_") .append(symbol); mcld::StringEntry<llvm::StringRef>* from_real = script_.renameMap().insert(from_real_str, exist); from_real->setValue(symbol); if (exist) mcld::warning(mcld::diag::rewrap) << symbol << from_real_str; } // --portalbe=symbol for (llvm::opt::Arg* arg : args.filtered(kOpt_Portable)) { bool exist = false; const char* symbol = arg->getValue(); // symbol -> symbol_portable mcld::StringEntry<llvm::StringRef>* to_wrap = script_.renameMap().insert(symbol, exist); std::string to_wrap_str; to_wrap_str.append(symbol) .append("_portable"); to_wrap->setValue(to_wrap_str); if (exist) mcld::warning(mcld::diag::rewrap) << symbol << to_wrap_str; // __real_symbol -> symbol std::string from_real_str; to_wrap_str.append("__real_") .append(symbol); mcld::StringEntry<llvm::StringRef>* from_real = script_.renameMap().insert(from_real_str, exist); from_real->setValue(symbol); if (exist) mcld::warning(mcld::diag::rewrap) << symbol << from_real_str; } // --section-start=section=addr for (llvm::opt::Arg* arg : args.filtered(kOpt_SectionStart)) { llvm::StringRef value = arg->getValue(); const size_t pos = value.find('='); uint64_t addr = 0; value.substr(pos + 1).getAsInteger(0, addr); bool exist = false; mcld::StringEntry<uint64_t>* mapping = script_.addressMap().insert(value.substr(0, pos), exist); mapping->setValue(addr); } // -Tbss=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Tbss)) { llvm::StringRef value = arg->getValue(); uint64_t addr = 0; if (value.getAsInteger(0, addr)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } bool exist = false; mcld::StringEntry<uint64_t>* mapping = script_.addressMap().insert(".bss", exist); mapping->setValue(addr); } // -Tdata=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Tdata)) { llvm::StringRef value = arg->getValue(); uint64_t addr = 0; if (value.getAsInteger(0, addr)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } bool exist = false; mcld::StringEntry<uint64_t>* mapping = script_.addressMap().insert(".data", exist); mapping->setValue(addr); } // -Ttext=value if (llvm::opt::Arg* arg = args.getLastArg(kOpt_Ttext)) { llvm::StringRef value = arg->getValue(); uint64_t addr = 0; if (value.getAsInteger(0, addr)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } bool exist = false; mcld::StringEntry<uint64_t>* mapping = script_.addressMap().insert(".text", exist); mapping->setValue(addr); } //===--------------------------------------------------------------------===// // Optimization //===--------------------------------------------------------------------===// // --[no-]gc-sections if (llvm::opt::Arg* arg = args.getLastArg(kOpt_GCSections, kOpt_NoGCSections)) { if (arg->getOption().matches(kOpt_GCSections)) { config_.options().setGCSections(true); } else { config_.options().setGCSections(false); } } // --[no-]print-gc-sections if (llvm::opt::Arg* arg = args.getLastArg(kOpt_PrintGCSections, kOpt_NoPrintGCSections)) { if (arg->getOption().matches(kOpt_PrintGCSections)) { config_.options().setPrintGCSections(true); } else { config_.options().setPrintGCSections(false); } } // --[no-]ld-generated-unwind-info if (llvm::opt::Arg* arg = args.getLastArg(kOpt_LDGeneratedUnwindInfo, kOpt_NoLDGeneratedUnwindInfo)) { if (arg->getOption().matches(kOpt_LDGeneratedUnwindInfo)) { config_.options().setGenUnwindInfo(true); } else { config_.options().setGenUnwindInfo(false); } } // --icf=mode if (llvm::opt::Arg* arg = args.getLastArg(kOpt_ICF)) { mcld::GeneralOptions::ICF mode = llvm::StringSwitch<mcld::GeneralOptions::ICF>(arg->getValue()) .Case("none", mcld::GeneralOptions::ICF::None) .Case("all", mcld::GeneralOptions::ICF::All) .Case("safe", mcld::GeneralOptions::ICF::Safe) .Default(mcld::GeneralOptions::ICF::Unknown); if (mode == mcld::GeneralOptions::ICF::Unknown) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } config_.options().setICFMode(mode); } // --icf-iterations if (llvm::opt::Arg* arg = args.getLastArg(kOpt_ICFIters)) { llvm::StringRef value = arg->getValue(); int num; if (value.getAsInteger(0, num) || (num < 0)) { mcld::errs() << "Invalid value for" << arg->getOption().getPrefixedName() << ": " << arg->getValue() << "\n"; return false; } config_.options().setICFIterations(num); } // --[no-]print-icf-sections if (llvm::opt::Arg* arg = args.getLastArg(kOpt_PrintICFSections, kOpt_NoPrintICFSections)) { if (arg->getOption().matches(kOpt_PrintICFSections)) { config_.options().setPrintICFSections(true); } else { config_.options().setPrintICFSections(false); } } //===--------------------------------------------------------------------===// // Positional //===--------------------------------------------------------------------===// // # of regular objects, script, and namespec. size_t input_num = 0; typedef std::unique_ptr<mcld::InputAction> Action; std::vector<Action> actions; Action action; actions.reserve(32); for (llvm::opt::Arg* arg : args) { const unsigned index = arg->getIndex(); switch (arg->getOption().getID()) { // -T script case kOpt_Script: { const char* value = arg->getValue(); config_.options().getScriptList().push_back(value); // FIXME: Let index of script file be 0. action.reset(new mcld::ScriptAction( 0x0, value, mcld::ScriptFile::LDScript, script_.directories())); actions.push_back(std::move(action)); action.reset(new mcld::ContextAction(0x0)); actions.push_back(std::move(action)); action.reset(new mcld::MemoryAreaAction(0x0, mcld::FileHandle::ReadOnly)); actions.push_back(std::move(action)); ++input_num; break; } // --defsym=symbol=expr case kOpt_DefSym: { std::string expr; expr.append(arg->getValue()) .append(";"); script_.defsyms().push_back(std::move(expr)); action.reset(new mcld::DefSymAction(index, script_.defsyms().back())); actions.push_back(std::move(action)); break; } // -l namespec case kOpt_Namespec: { action.reset(new mcld::NamespecAction( index, arg->getValue(), script_.directories())); actions.push_back(std::move(action)); action.reset(new mcld::ContextAction(index)); actions.push_back(std::move(action)); action.reset(new mcld::MemoryAreaAction(index, mcld::FileHandle::ReadOnly)); actions.push_back(std::move(action)); ++input_num; break; } // --whole-archive case kOpt_WholeArchive: { action.reset(new mcld::WholeArchiveAction(index)); actions.push_back(std::move(action)); break; } // --no-whole-archive case kOpt_NoWholeArchive: { action.reset(new mcld::NoWholeArchiveAction(index)); actions.push_back(std::move(action)); break; } // --as-needed case kOpt_AsNeeded: { action.reset(new mcld::AsNeededAction(index)); actions.push_back(std::move(action)); break; } // --no-as-needed case kOpt_NoAsNeeded: { action.reset(new mcld::NoAsNeededAction(index)); actions.push_back(std::move(action)); break; } // --add-needed // FIXME: This is deprecated. Should be --copy-dt-needed-entries. case kOpt_AddNeeded: case kOpt_CopyDTNeeded: { action.reset(new mcld::AddNeededAction(index)); actions.push_back(std::move(action)); break; } // --no-add-needed // FIXME: This is deprecated. Should be --no-copy-dt-needed-entries. case kOpt_NoAddNeeded: case kOpt_NoCopyDTNeeded: { action.reset(new mcld::AddNeededAction(index)); actions.push_back(std::move(action)); break; } // -Bdynamic case kOpt_Bdynamic: { action.reset(new mcld::BDynamicAction(index)); actions.push_back(std::move(action)); break; } // -Bstatic case kOpt_Bstatic: { action.reset(new mcld::BStaticAction(index)); actions.push_back(std::move(action)); break; } // --start-group case kOpt_StartGroup: { action.reset(new mcld::StartGroupAction(index)); actions.push_back(std::move(action)); break; } // --end-group case kOpt_EndGroup: { action.reset(new mcld::EndGroupAction(index)); actions.push_back(std::move(action)); break; } case kOpt_INPUT: { action.reset(new mcld::InputFileAction(index, arg->getValue())); actions.push_back(std::move(action)); action.reset(new mcld::ContextAction(index)); actions.push_back(std::move(action)); action.reset(new mcld::MemoryAreaAction(index, mcld::FileHandle::ReadOnly)); actions.push_back(std::move(action)); ++input_num; break; } default: break; } } if (input_num == 0) { mcld::fatal(mcld::diag::err_no_inputs); return false; } // Stable sort std::stable_sort(actions.begin(), actions.end(), [] (const Action& X, const Action& Y) { return X->position() < Y->position(); }); if (!InitializeInputs(ir_builder_, actions)) { mcld::errs() << "Failed to initialize input tree!\n"; return false; } //===--------------------------------------------------------------------===// // Unknown //===--------------------------------------------------------------------===// std::vector<std::string> unknown_args = args.getAllArgValues(kOpt_UNKNOWN); for (std::string arg : unknown_args) mcld::warning(mcld::diag::warn_unsupported_option) << arg; return true; } std::unique_ptr<Driver> Driver::Create(llvm::ArrayRef<const char*> argv) { // Parse command line options. OptTable opt_table; unsigned missing_arg_idx; unsigned missing_arg_count; llvm::opt::InputArgList args = opt_table.ParseArgs(argv.slice(1), missing_arg_idx, missing_arg_count); if (missing_arg_count > 0) { mcld::errs() << "Argument to '" << args.getArgString(missing_arg_idx) << "' is missing (expected " << missing_arg_count << ((missing_arg_count > 1) ? " values" : " value") << ")\n"; return nullptr; } std::unique_ptr<Driver> result(new Driver(argv[0])); // Return quickly if -help is specified. if (args.hasArg(kOpt_Help)) { opt_table.PrintHelp(mcld::outs(), argv[0], "MCLinker", /* FlagsToInclude */0, /* FlagsToExclude */0); return nullptr; } // Print version information if requested. if (args.hasArg(kOpt_Version)) { mcld::outs() << result->config_.options().getVersionString() << "\n"; } // Setup instance from arguments. if (!result->TranslateArguments(args)) { return nullptr; } return result; } bool Driver::Run() { mcld::Initialize(); if (!linker_.emulate(script_, config_)) { mcld::errs() << "Failed to emulate target!\n"; return false; } if (!linker_.link(module_, ir_builder_)) { mcld::errs() << "Failed to link objects!\n"; return false; } if (!linker_.emit(module_, module_.name())) { mcld::errs() << "Failed to emit output!\n"; return false; } mcld::Finalize(); return true; } } // anonymous namespace int main(int argc, char** argv) { std::unique_ptr<Driver> driver = Driver::Create(llvm::makeArrayRef(argv, argc)); if ((driver == nullptr) || !driver->Run()) { return EXIT_FAILURE; } else { return EXIT_SUCCESS; } }