blob: 375dd1ad7048e266dd9bc7c29e6fa787d15a7bb6 [file] [log] [blame]
Martell Malone894dbbe2017-09-11 17:02:591//===- MinGW/Driver.cpp ---------------------------------------------------===//
2//
3// The LLVM Linker
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9///
10/// GNU ld style linker driver for COFF currently supporting mingw-w64.
11///
12//===----------------------------------------------------------------------===//
13
Rui Ueyama6cb71412017-10-02 22:05:4614#include "lld/Common/Driver.h"
Martell Malone894dbbe2017-09-11 17:02:5915#include "llvm/ADT/ArrayRef.h"
16#include "llvm/ADT/Optional.h"
Martin Storsjoa79762a2017-09-11 20:43:3917#include "llvm/ADT/StringExtras.h"
Martell Malone894dbbe2017-09-11 17:02:5918#include "llvm/ADT/StringRef.h"
Martell Malone894dbbe2017-09-11 17:02:5919#include "llvm/Option/Arg.h"
20#include "llvm/Option/ArgList.h"
21#include "llvm/Option/Option.h"
22#include "llvm/Support/CommandLine.h"
23#include "llvm/Support/FileSystem.h"
24#include "llvm/Support/Path.h"
25
26#if !defined(_MSC_VER) && !defined(__MINGW32__)
27#include <unistd.h>
28#endif
29
30using namespace lld;
31using namespace llvm;
32
Rui Ueyamad2da71f2017-09-11 22:04:3733LLVM_ATTRIBUTE_NORETURN static void error(const Twine &Msg) {
34 errs() << Msg << "\n";
35 exit(1);
36}
37
Martell Malone894dbbe2017-09-11 17:02:5938// Create OptTable
39enum {
40 OPT_INVALID = 0,
41#define OPTION(_1, _2, ID, _4, _5, _6, _7, _8, _9, _10, _11, _12) OPT_##ID,
42#include "Options.inc"
43#undef OPTION
44};
45
46// Create prefix string literals used in Options.td
Rui Ueyamac16fd252017-09-11 22:04:1347#define PREFIX(NAME, VALUE) static const char *const NAME[] = VALUE;
Martell Malone894dbbe2017-09-11 17:02:5948#include "Options.inc"
49#undef PREFIX
50
51// Create table mapping all options defined in Options.td
52static const opt::OptTable::Info InfoTable[] = {
53#define OPTION(X1, X2, ID, KIND, GROUP, ALIAS, X7, X8, X9, X10, X11, X12) \
54 {X1, X2, X10, X11, OPT_##ID, opt::Option::KIND##Class, \
55 X9, X8, OPT_##GROUP, OPT_##ALIAS, X7, X12},
56#include "Options.inc"
57#undef OPTION
58};
59
Rui Ueyamac16fd252017-09-11 22:04:1360namespace {
Rui Ueyama44747962017-09-11 22:04:2561class MinGWOptTable : public opt::OptTable {
Martell Malone894dbbe2017-09-11 17:02:5962public:
Rui Ueyama44747962017-09-11 22:04:2563 MinGWOptTable() : OptTable(InfoTable, false) {}
Martell Malone894dbbe2017-09-11 17:02:5964 opt::InputArgList parse(ArrayRef<const char *> Argv);
65};
Martell Malone894dbbe2017-09-11 17:02:5966} // namespace
67
Rui Ueyamad2da71f2017-09-11 22:04:3768opt::InputArgList MinGWOptTable::parse(ArrayRef<const char *> Argv) {
69 unsigned MissingIndex;
70 unsigned MissingCount;
71
72 SmallVector<const char *, 256> Vec(Argv.data(), Argv.data() + Argv.size());
73 opt::InputArgList Args = this->ParseArgs(Vec, MissingIndex, MissingCount);
74
75 if (MissingCount)
76 error(StringRef(Args.getArgString(MissingIndex)) + ": missing argument");
77 for (auto *Arg : Args.filtered(OPT_UNKNOWN))
78 error("unknown argument: " + Arg->getSpelling());
Rui Ueyamaa835bab2017-09-13 20:30:5979 if (!Args.hasArg(OPT_INPUT) && !Args.hasArg(OPT_l))
Rui Ueyamad2da71f2017-09-11 22:04:3780 error("no input files");
81 return Args;
Martell Malone894dbbe2017-09-11 17:02:5982}
83
84// Find a file by concatenating given paths.
85static Optional<std::string> findFile(StringRef Path1, const Twine &Path2) {
86 SmallString<128> S;
87 sys::path::append(S, Path1, Path2);
88 if (sys::fs::exists(S))
89 return S.str().str();
90 return None;
91}
92
Martell Malone894dbbe2017-09-11 17:02:5993// This is for -lfoo. We'll look for libfoo.dll.a or libfoo.a from search paths.
Rui Ueyama5941b0b2017-09-11 20:14:4794static std::string
95searchLibrary(StringRef Name, ArrayRef<StringRef> SearchPaths, bool BStatic) {
96 if (Name.startswith(":")) {
97 for (StringRef Dir : SearchPaths)
98 if (Optional<std::string> S = findFile(Dir, Name.substr(1)))
99 return *S;
Martell Malone894dbbe2017-09-11 17:02:59100 error("unable to find library -l" + Name);
Martell Malone894dbbe2017-09-11 17:02:59101 }
Martell Malone894dbbe2017-09-11 17:02:59102
Rui Ueyama5941b0b2017-09-11 20:14:47103 for (StringRef Dir : SearchPaths) {
104 if (!BStatic)
105 if (Optional<std::string> S = findFile(Dir, "lib" + Name + ".dll.a"))
106 return *S;
107 if (Optional<std::string> S = findFile(Dir, "lib" + Name + ".a"))
108 return *S;
Martell Malone894dbbe2017-09-11 17:02:59109 }
Rui Ueyama5941b0b2017-09-11 20:14:47110 error("unable to find library -l" + Name);
Martell Malone894dbbe2017-09-11 17:02:59111}
112
Rui Ueyama5941b0b2017-09-11 20:14:47113// Convert Unix-ish command line arguments to Windows-ish ones and
114// then call coff::link.
Rui Ueyama95114b42017-09-11 22:04:01115bool mingw::link(ArrayRef<const char *> ArgsArr, raw_ostream &Diag) {
Rui Ueyama44747962017-09-11 22:04:25116 MinGWOptTable Parser;
Martell Malone894dbbe2017-09-11 17:02:59117 opt::InputArgList Args = Parser.parse(ArgsArr.slice(1));
Martell Malone894dbbe2017-09-11 17:02:59118
Rui Ueyama5941b0b2017-09-11 20:14:47119 std::vector<std::string> LinkArgs;
120 auto Add = [&](const Twine &S) { LinkArgs.push_back(S.str()); };
Martell Malone894dbbe2017-09-11 17:02:59121
Martin Storsjoa79762a2017-09-11 20:43:39122 Add("lld-link");
Martin Storsjo31fe4cd2017-09-13 19:29:39123 Add("-lldmingw");
Martell Malone894dbbe2017-09-11 17:02:59124
Martin Storsjo6d8dace2017-09-12 19:23:54125 if (auto *A = Args.getLastArg(OPT_entry)) {
126 StringRef S = A->getValue();
127 if (Args.getLastArgValue(OPT_m) == "i386pe" && S.startswith("_"))
128 Add("-entry:" + S.substr(1));
129 else
130 Add("-entry:" + S);
131 }
132
Rui Ueyama5941b0b2017-09-11 20:14:47133 if (auto *A = Args.getLastArg(OPT_subs))
134 Add("-subsystem:" + StringRef(A->getValue()));
Rui Ueyamac3e46022017-09-11 20:54:51135 if (auto *A = Args.getLastArg(OPT_out_implib))
Rui Ueyama5941b0b2017-09-11 20:14:47136 Add("-implib:" + StringRef(A->getValue()));
137 if (auto *A = Args.getLastArg(OPT_stack))
138 Add("-stack:" + StringRef(A->getValue()));
Martell Malone894dbbe2017-09-11 17:02:59139
Rui Ueyama5941b0b2017-09-11 20:14:47140 if (auto *A = Args.getLastArg(OPT_o))
141 Add("-out:" + StringRef(A->getValue()));
142 else if (Args.hasArg(OPT_shared))
143 Add("-out:a.dll");
Martell Malone894dbbe2017-09-11 17:02:59144 else
Rui Ueyama5941b0b2017-09-11 20:14:47145 Add("-out:a.exe");
Martell Malone894dbbe2017-09-11 17:02:59146
Rui Ueyama5941b0b2017-09-11 20:14:47147 if (Args.hasArg(OPT_shared))
148 Add("-dll");
Rui Ueyama7d306a52017-09-13 19:40:07149 if (Args.hasArg(OPT_verbose))
150 Add("-verbose");
Rui Ueyama5941b0b2017-09-11 20:14:47151
152 if (auto *A = Args.getLastArg(OPT_m)) {
153 StringRef S = A->getValue();
154 if (S == "i386pe")
155 Add("-machine:x86");
156 else if (S == "i386pep")
157 Add("-machine:x64");
158 else if (S == "thumb2pe")
159 Add("-machine:arm");
160 else if (S == "arm64pe")
161 Add("-machine:arm64");
162 else
163 error("unknown parameter: -m" + S);
164 }
165
Martell Malone0d176382017-09-11 21:36:37166 for (auto *A : Args.filtered(OPT_mllvm))
167 Add("-mllvm:" + StringRef(A->getValue()));
168
Rui Ueyama5941b0b2017-09-11 20:14:47169 if (Args.getLastArgValue(OPT_m) == "i386pe")
170 Add("-alternatename:__image_base__=___ImageBase");
171 else
172 Add("-alternatename:__image_base__=__ImageBase");
173
174 std::vector<StringRef> SearchPaths;
175 for (auto *A : Args.filtered(OPT_L))
176 SearchPaths.push_back(A->getValue());
177
Martin Storsjo32e16262017-09-13 07:28:09178 StringRef Prefix = "";
Martin Storsjo2ba37d42017-09-13 19:29:44179 bool Static = false;
180 for (auto *A : Args) {
Rui Ueyama9c37d8c2017-09-13 19:44:36181 switch (A->getOption().getUnaliasedOption().getID()) {
Martin Storsjo32e16262017-09-13 07:28:09182 case OPT_INPUT:
Rui Ueyamab3592632017-09-14 18:33:09183 if (StringRef(A->getValue()).endswith_lower(".def"))
Martin Storsjo064b0fa2017-09-13 07:28:13184 Add("-def:" + StringRef(A->getValue()));
185 else
186 Add(Prefix + StringRef(A->getValue()));
Martin Storsjo32e16262017-09-13 07:28:09187 break;
188 case OPT_l:
Martin Storsjo2ba37d42017-09-13 19:29:44189 Add(Prefix + searchLibrary(A->getValue(), SearchPaths, Static));
Martin Storsjo32e16262017-09-13 07:28:09190 break;
191 case OPT_whole_archive:
192 Prefix = "-wholearchive:";
193 break;
194 case OPT_no_whole_archive:
195 Prefix = "";
196 break;
Martin Storsjo2ba37d42017-09-13 19:29:44197 case OPT_Bstatic:
198 Static = true;
199 break;
200 case OPT_Bdynamic:
201 Static = false;
202 break;
Martin Storsjo32e16262017-09-13 07:28:09203 }
Rui Ueyama5941b0b2017-09-11 20:14:47204 }
205
Martin Storsjoa79762a2017-09-11 20:43:39206 if (Args.hasArg(OPT_verbose) || Args.hasArg(OPT__HASH_HASH_HASH))
207 outs() << llvm::join(LinkArgs, " ") << "\n";
208
209 if (Args.hasArg(OPT__HASH_HASH_HASH))
210 return true;
211
Rui Ueyama5941b0b2017-09-11 20:14:47212 // Repack vector of strings to vector of const char pointers for coff::link.
Martell Malone894dbbe2017-09-11 17:02:59213 std::vector<const char *> Vec;
214 for (const std::string &S : LinkArgs)
215 Vec.push_back(S.c_str());
216 return coff::link(Vec);
217}