#include "lld/Common/CommonLinkerContext.h"
#include "lld/Common/Driver.h"
#include "lld/Common/ErrorHandler.h"
#include "lld/Common/Memory.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/ADT/StringSwitch.h"
#include "llvm/ADT/Twine.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/CrashRecoveryContext.h"
#include "llvm/Support/Path.h"
#include "llvm/Support/Process.h"
#include "llvm/TargetParser/Host.h"
#include "llvm/TargetParser/Triple.h"
#include <cstdlib>
using namespace lld;
using namespace llvm;
using namespace llvm::sys;
static void err(const Twine &s) { llvm::errs() << s << "\n"; }
static Flavor getFlavor(StringRef s) {
return StringSwitch<Flavor>(s)
.CasesLower("ld", "ld.lld", "gnu", Gnu)
.CasesLower("wasm", "ld-wasm", Wasm)
.CaseLower("link", WinLink)
.CasesLower("ld64", "ld64.lld", "darwin", Darwin)
.Default(Invalid);
}
static cl::TokenizerCallback getDefaultQuotingStyle() {
if (Triple(sys::getProcessTriple()).getOS() == Triple::Win32)
return cl::TokenizeWindowsCommandLine;
return cl::TokenizeGNUCommandLine;
}
static bool isPETargetName(StringRef s) {
return s == "i386pe" || s == "i386pep" || s == "thumb2pe" || s == "arm64pe" ||
s == "arm64ecpe";
}
static std::optional<bool> isPETarget(llvm::ArrayRef<const char *> args) {
for (auto it = args.begin(); it + 1 != args.end(); ++it) {
if (StringRef(*it) != "-m")
continue;
return isPETargetName(*(it + 1));
}
SmallVector<const char *, 256> expandedArgs(args.data(),
args.data() + args.size());
BumpPtrAllocator a;
StringSaver saver(a);
cl::ExpansionContext ectx(saver.getAllocator(), getDefaultQuotingStyle());
if (Error e = ectx.expandResponseFiles(expandedArgs)) {
err(toString(std::move(e)));
return std::nullopt;
}
for (auto it = expandedArgs.begin(); it + 1 != expandedArgs.end(); ++it) {
if (StringRef(*it) != "-m")
continue;
return isPETargetName(*(it + 1));
}
#ifdef LLD_DEFAULT_LD_LLD_IS_MINGW
return true;
#else
return false;
#endif
}
static Flavor parseProgname(StringRef progname) {
if (progname == "ld")
return Gnu;
SmallVector<StringRef, 3> v;
progname.split(v, "-");
for (StringRef s : v)
if (Flavor f = getFlavor(s))
return f;
return Invalid;
}
static Flavor
parseFlavorWithoutMinGW(llvm::SmallVectorImpl<const char *> &argsV) {
if (argsV.size() > 1 && argsV[1] == StringRef("-flavor")) {
if (argsV.size() <= 2) {
err("missing arg value for '-flavor'");
return Invalid;
}
Flavor f = getFlavor(argsV[2]);
if (f == Invalid) {
err("Unknown flavor: " + StringRef(argsV[2]));
return Invalid;
}
argsV.erase(argsV.begin() + 1, argsV.begin() + 3);
return f;
}
StringRef arg0 = path::filename(argsV[0]);
if (arg0.ends_with_insensitive(".exe"))
arg0 = arg0.drop_back(4);
Flavor f = parseProgname(arg0);
if (f == Invalid) {
err("lld is a generic driver.\n"
"Invoke ld.lld (Unix), ld64.lld (macOS), lld-link (Windows), wasm-ld"
" (WebAssembly) instead");
return Invalid;
}
return f;
}
static Flavor parseFlavor(llvm::SmallVectorImpl<const char *> &argsV) {
Flavor f = parseFlavorWithoutMinGW(argsV);
if (f == Gnu) {
auto isPE = isPETarget(argsV);
if (!isPE)
return Invalid;
if (*isPE)
return MinGW;
}
return f;
}
static Driver whichDriver(llvm::SmallVectorImpl<const char *> &argsV,
llvm::ArrayRef<DriverDef> drivers) {
Flavor f = parseFlavor(argsV);
auto it =
llvm::find_if(drivers, [=](auto &driverdef) { return driverdef.f == f; });
if (it == drivers.end()) {
return [](llvm::ArrayRef<const char *>, llvm::raw_ostream &,
llvm::raw_ostream &, bool, bool) { return false; };
}
return it->d;
}
namespace lld {
bool inTestOutputDisabled = false;
int unsafeLldMain(llvm::ArrayRef<const char *> args,
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS,
llvm::ArrayRef<DriverDef> drivers, bool exitEarly) {
SmallVector<const char *, 256> argsV(args);
Driver d = whichDriver(argsV, drivers);
int r = !d(argsV, stdoutOS, stderrOS, exitEarly, inTestOutputDisabled);
if (exitEarly)
exitLld(r);
CommonLinkerContext::destroy();
return r;
}
}
Result lld::lldMain(llvm::ArrayRef<const char *> args,
llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS,
llvm::ArrayRef<DriverDef> drivers) {
int r = 0;
{
llvm::CrashRecoveryContext crc;
if (!crc.RunSafely([&]() {
r = unsafeLldMain(args, stdoutOS, stderrOS, drivers,
false);
}))
return {crc.RetCode, false};
}
llvm::CrashRecoveryContext crc;
if (!crc.RunSafely([&]() { CommonLinkerContext::destroy(); })) {
return {r, false};
}
return {r, true};
}