summaryrefslogtreecommitdiffstats
path: root/lld/lib/Driver/UniversalDriver.cpp
diff options
context:
space:
mode:
authorMichael J. Spencer <bigcheesegs@gmail.com>2013-04-04 22:04:16 +0000
committerMichael J. Spencer <bigcheesegs@gmail.com>2013-04-04 22:04:16 +0000
commitdfe85483d58170a5146990c3987e61b45bea413f (patch)
tree67aa4e666efb34ea97591cb2362e7c674e6a11e2 /lld/lib/Driver/UniversalDriver.cpp
parentc451e5766e8b27f7ca55614364db0b101b729121 (diff)
downloadbcm5719-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.cpp144
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
OpenPOWER on IntegriCloud