diff options
| author | Michael J. Spencer <bigcheesegs@gmail.com> | 2013-04-04 22:04:16 +0000 |
|---|---|---|
| committer | Michael J. Spencer <bigcheesegs@gmail.com> | 2013-04-04 22:04:16 +0000 |
| commit | dfe85483d58170a5146990c3987e61b45bea413f (patch) | |
| tree | 67aa4e666efb34ea97591cb2362e7c674e6a11e2 /lld/lib/Driver/UniversalDriver.cpp | |
| parent | c451e5766e8b27f7ca55614364db0b101b729121 (diff) | |
| download | bcm5719-llvm-dfe85483d58170a5146990c3987e61b45bea413f.tar.gz bcm5719-llvm-dfe85483d58170a5146990c3987e61b45bea413f.zip | |
[Driver] Fix symlinked universal driver behavior and add a test.
llvm-svn: 178798
Diffstat (limited to 'lld/lib/Driver/UniversalDriver.cpp')
| -rw-r--r-- | lld/lib/Driver/UniversalDriver.cpp | 144 |
1 files changed, 84 insertions, 60 deletions
diff --git a/lld/lib/Driver/UniversalDriver.cpp b/lld/lib/Driver/UniversalDriver.cpp index 5164f9da69c..001c354d309 100644 --- a/lld/lib/Driver/UniversalDriver.cpp +++ b/lld/lib/Driver/UniversalDriver.cpp @@ -14,96 +14,120 @@ //===----------------------------------------------------------------------===// #include "lld/Driver/Driver.h" -#include "lld/ReaderWriter/MachOTargetInfo.h" -#include "llvm/ADT/ArrayRef.h" +#include "lld/Core/LLVM.h" + #include "llvm/ADT/StringExtras.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/ADT/StringSwitch.h" #include "llvm/Support/Host.h" -#include "llvm/Support/ManagedStatic.h" #include "llvm/Support/Path.h" #include "llvm/Support/raw_ostream.h" -#include "llvm/Support/Signals.h" - -#include <memory> -namespace lld { +using namespace lld; +namespace { +enum class Flavor { + invalid, + gnu_ld, // -flavor gnu + win_link, // -flavor link + darwin_ld, // -flavor darwin + core // -flavor core OR -core +}; -bool UniversalDriver::link(int argc, const char *argv[]) { - // Convert argv[] C-array to vector. - std::vector<const char *> args; - args.assign(&argv[0], &argv[argc]); - - // Determine flavor of link based on command name or -flavor argument. - // Note: 'args' is modified to remove -flavor option. - Flavor flavor = selectFlavor(args); - - // Switch to appropriate driver. - switch (flavor) { - case Flavor::gnu_ld: - return GnuLdDriver::linkELF(args.size(), &args[0]); - case Flavor::darwin_ld: - return DarwinLdDriver::linkMachO(args.size(), &args[0]); - case Flavor::core: - return CoreDriver::link(args.size(), &args[0]); - case Flavor::win_link: - llvm_unreachable("Unsupported flavor"); - case Flavor::invalid: - return true; - } +Flavor strToFlavor(StringRef str) { + return llvm::StringSwitch<Flavor>(str) + .Case("gnu", Flavor::gnu_ld) + .Case("link", Flavor::win_link) + .Case("darwin", Flavor::darwin_ld) + .Case("core", Flavor::core) + .Case("ld", Flavor::gnu_ld) // deprecated + .Default(Flavor::invalid); } +struct ProgramNameParts { + StringRef _target; + StringRef _flavor; +}; + +ProgramNameParts parseProgramName(StringRef programName) { + SmallVector<StringRef, 3> components; + llvm::SplitString(programName, components, "-"); + ProgramNameParts ret; + + using std::begin; + using std::end; + + // Erase any lld components. + components.erase(std::remove(components.begin(), components.end(), "lld"), + components.end()); + + // Find the flavor component. + auto flIter = std::find_if(components.begin(), components.end(), + [](StringRef str)->bool { + return strToFlavor(str) != Flavor::invalid; + }); + if (flIter != components.end()) { + ret._flavor = *flIter; + components.erase(flIter); + } + // Any remaining component must be the target. + if (components.size() == 1) + ret._target = components[0]; + return ret; +} -/// Pick the flavor of driver to use based on the command line and -/// host environment. -UniversalDriver::Flavor UniversalDriver::selectFlavor( - std::vector<const char*> &args) { +Flavor selectFlavor(std::vector<const char *> &args, raw_ostream &diag) { // -core as first arg is shorthand for -flavor core. - if (args.size() >= 1 && StringRef(args[1]) == "-core") { + if (args.size() > 1 && StringRef(args[1]) == "-core") { args.erase(args.begin() + 1); return Flavor::core; } // Handle -flavor as first arg. - if (args.size() >= 2 && StringRef(args[1]) == "-flavor") { + if (args.size() > 2 && StringRef(args[1]) == "-flavor") { Flavor flavor = strToFlavor(args[2]); args.erase(args.begin() + 1); args.erase(args.begin() + 1); if (flavor == Flavor::invalid) - llvm::errs() << "error: '" << args[2] << "' invalid value for -flavor.\n"; + diag << "error: '" << args[2] << "' invalid value for -flavor.\n"; return flavor; } - // Check if flavor is at end of program name (e.g. "lld-gnu"); - SmallVector<StringRef, 3> components; - llvm::SplitString(args[0], components, "-"); - Flavor flavor = strToFlavor(components.back()); - + Flavor flavor = + strToFlavor(parseProgramName(llvm::sys::path::stem(args[0]))._flavor); + // If flavor still undetermined, then error out. if (flavor == Flavor::invalid) - llvm::errs() << "error: failed to determine driver flavor from program name" - " '" << args[0] << "'.\n"; + diag << "error: failed to determine driver flavor from program name" + " '" << args[0] << "'.\n"; return flavor; } - -/// Maps flavor strings to Flavor enum values. -UniversalDriver::Flavor UniversalDriver::strToFlavor(StringRef str) { - return llvm::StringSwitch<Flavor>(str) - .Case("gnu", Flavor::gnu_ld) - .Case("darwin", Flavor::darwin_ld) - .Case("link", Flavor::win_link) - .Case("core", Flavor::core) - .Case("ld", Flavor::gnu_ld) // deprecated - .Default(Flavor::invalid); } +namespace lld { +bool UniversalDriver::link(int argc, const char *argv[], + raw_ostream &diagnostics) { + // Convert argv[] C-array to vector. + std::vector<const char *> args(argv, argv + argc); + + // Determine flavor of link based on command name or -flavor argument. + // Note: 'args' is modified to remove -flavor option. + Flavor flavor = selectFlavor(args, diagnostics); -} // namespace lld + // Switch to appropriate driver. + switch (flavor) { + case Flavor::gnu_ld: + return GnuLdDriver::linkELF(args.size(), args.data(), diagnostics); + case Flavor::darwin_ld: + return DarwinLdDriver::linkMachO(args.size(), args.data(), diagnostics); + case Flavor::core: + return CoreDriver::link(args.size(), args.data(), diagnostics); + case Flavor::win_link: + llvm_unreachable("Unsupported flavor"); + case Flavor::invalid: + return true; + } +} +} // end namespace lld |

