//===- lib/Driver/WinLinkDriver.cpp ---------------------------------------===// // // The LLVM Linker // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// /// /// \file /// /// Concrete instance of the Driver for Windows link.exe. /// //===----------------------------------------------------------------------===// #include "llvm/ADT/StringSwitch.h" #include "llvm/Option/Arg.h" #include "llvm/Option/Option.h" #include "llvm/Support/PathV2.h" #include "lld/Driver/Driver.h" #include "lld/ReaderWriter/PECOFFTargetInfo.h" namespace lld { namespace { // Create enum with OPT_xxx values for each option in WinLinkOptions.td enum WinLinkOpt { OPT_INVALID = 0, #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, FLAGS, PARAM, HELP, META) \ OPT_##ID, #include "WinLinkOptions.inc" LastOption #undef OPTION }; // Create prefix string literals used in WinLinkOptions.td #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE; #include "WinLinkOptions.inc" #undef PREFIX // Create table mapping all options defined in WinLinkOptions.td static const llvm::opt::OptTable::Info infoTable[] = { #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, FLAGS, PARAM, \ HELPTEXT, METAVAR) \ { PREFIX, NAME, HELPTEXT, METAVAR, OPT_##ID, llvm::opt::Option::KIND##Class, \ PARAM, FLAGS, OPT_##GROUP, OPT_##ALIAS }, #include "WinLinkOptions.inc" #undef OPTION }; // Create OptTable class for parsing actual command line arguments class WinLinkOptTable : public llvm::opt::OptTable { public: WinLinkOptTable() : OptTable(infoTable, llvm::array_lengthof(infoTable)){} }; // Returns the index of "--" or -1 if not found. int findDoubleDash(int argc, const char *argv[]) { for (int i = 0; i < argc; ++i) if (std::strcmp(argv[i], "--") == 0) return i; return -1; } // Parses -subsystem command line option. llvm::COFF::WindowsSubsystem strToWinSubsystem(std::string str) { std::string arg(StringRef(str).lower()); return llvm::StringSwitch(arg) .Case("windows", llvm::COFF::IMAGE_SUBSYSTEM_WINDOWS_GUI) .Case("console", llvm::COFF::IMAGE_SUBSYSTEM_WINDOWS_CUI) .Default(llvm::COFF::IMAGE_SUBSYSTEM_UNKNOWN); } // Add ".obj" extension if the given path name has no file extension. std::string canonicalizeInputFileName(std::string path) { if (llvm::sys::path::extension(path).empty()) return path.append(".obj"); return path; } // Replace a file extension with ".exe". If the given file has no // extension, just add ".exe". std::string getDefaultOutputFileName(std::string path) { StringRef ext = llvm::sys::path::extension(path); if (!ext.empty()) path.erase(path.size() - ext.size()); return path.append(".exe"); } } // namespace bool WinLinkDriver::linkPECOFF(int argc, const char *argv[], raw_ostream &diagnostics) { PECOFFTargetInfo info; if (parse(argc, argv, info, diagnostics)) return true; return link(info, diagnostics); } bool WinLinkDriver::parse(int argc, const char *argv[], PECOFFTargetInfo &info, raw_ostream &diagnostics) { // Arguments after "--" are interpreted as filenames even if they start with // a hyphen or a slash. This is not compatible with link.exe but useful for // us to test lld on Unix. int doubleDashPosition = findDoubleDash(argc, argv); int argEnd = (doubleDashPosition > 0) ? doubleDashPosition : argc; // Parse command line options using WinLinkOptions.td std::unique_ptr parsedArgs; WinLinkOptTable table; unsigned missingIndex; unsigned missingCount; parsedArgs.reset( table.ParseArgs(&argv[1], &argv[argEnd], missingIndex, missingCount)); if (missingCount) { diagnostics << "error: missing arg value for '" << parsedArgs->getArgString(missingIndex) << "' expected " << missingCount << " argument(s).\n"; return true; } // Handle -help if (parsedArgs->getLastArg(OPT_help)) { table.PrintHelp(llvm::outs(), argv[0], "LLVM Linker", false); return true; } // Show warning for unknown arguments for (auto it = parsedArgs->filtered_begin(OPT_UNKNOWN), ie = parsedArgs->filtered_end(); it != ie; ++it) { diagnostics << "warning: ignoring unknown argument: " << (*it)->getAsString(*parsedArgs) << "\n"; } // Copy -mllvm for (llvm::opt::arg_iterator it = parsedArgs->filtered_begin(OPT_mllvm), ie = parsedArgs->filtered_end(); it != ie; ++it) { info.appendLLVMOption((*it)->getValue()); } // Handle -subsystem if (llvm::opt::Arg *arg = parsedArgs->getLastArg(OPT_subsystem)) { llvm::COFF::WindowsSubsystem subsystem = strToWinSubsystem(arg->getValue()); if (subsystem == llvm::COFF::IMAGE_SUBSYSTEM_UNKNOWN) { diagnostics << "error: unknown subsystem name: " << arg->getValue() << "\n"; return true; } info.setSubsystem(subsystem); } // Hanlde -out if (llvm::opt::Arg *outpath = parsedArgs->getLastArg(OPT_out)) info.setOutputPath(outpath->getValue()); // Add input files std::vector inputPaths; for (llvm::opt::arg_iterator it = parsedArgs->filtered_begin(OPT_INPUT), ie = parsedArgs->filtered_end(); it != ie; ++it) { inputPaths.push_back((*it)->getValue()); } // Arguments after "--" are also input files if (doubleDashPosition > 0) for (int i = doubleDashPosition + 1; i < argc; ++i) inputPaths.push_back(argv[i]); // Add ".obj" extension for those who have no file extension. for (const StringRef &path : inputPaths) info.appendInputFile(canonicalizeInputFileName(path)); // If -out option was not specified, the default output file name is // constructed by replacing an extension with ".exe". if (info.outputPath().empty() && !inputPaths.empty()) info.setOutputPath(getDefaultOutputFileName(inputPaths[0])); // Validate the combination of options used. return info.validate(diagnostics); } } // namespace lld