blob: 44c12520b98776fac4006568d4fd7080d4b03a2e [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()));
Martin Storsjoa84a4772017-10-12 05:37:18139 if (auto *A = Args.getLastArg(OPT_output_def))
140 Add("-output-def:" + StringRef(A->getValue()));
Martell Malone894dbbe2017-09-11 17:02:59141
Rui Ueyama5941b0b2017-09-11 20:14:47142 if (auto *A = Args.getLastArg(OPT_o))
143 Add("-out:" + StringRef(A->getValue()));
144 else if (Args.hasArg(OPT_shared))
145 Add("-out:a.dll");
Martell Malone894dbbe2017-09-11 17:02:59146 else
Rui Ueyama5941b0b2017-09-11 20:14:47147 Add("-out:a.exe");
Martell Malone894dbbe2017-09-11 17:02:59148
Rui Ueyama5941b0b2017-09-11 20:14:47149 if (Args.hasArg(OPT_shared))
150 Add("-dll");
Rui Ueyama7d306a52017-09-13 19:40:07151 if (Args.hasArg(OPT_verbose))
152 Add("-verbose");
Martin Storsjoa84a4772017-10-12 05:37:18153 if (Args.hasArg(OPT_export_all_symbols))
154 Add("-export-all-symbols");
Martin Storsjo068512d2017-11-03 22:10:37155 if (!Args.hasArg(OPT_strip_all))
156 Add("-debug:dwarf");
Martin Storsjo3c8d4b52017-11-15 08:18:06157 if (Args.hasArg(OPT_large_address_aware))
158 Add("-largeaddressaware");
Rui Ueyama5941b0b2017-09-11 20:14:47159
Martin Storsjo094d8c02017-11-15 08:18:11160 Add(Args.hasArg(OPT_dynamicbase) ? "-dynamicbase" : "-dynamicbase:no");
161
Rui Ueyama5941b0b2017-09-11 20:14:47162 if (auto *A = Args.getLastArg(OPT_m)) {
163 StringRef S = A->getValue();
164 if (S == "i386pe")
165 Add("-machine:x86");
166 else if (S == "i386pep")
167 Add("-machine:x64");
168 else if (S == "thumb2pe")
169 Add("-machine:arm");
170 else if (S == "arm64pe")
171 Add("-machine:arm64");
172 else
173 error("unknown parameter: -m" + S);
174 }
175
Martell Malone0d176382017-09-11 21:36:37176 for (auto *A : Args.filtered(OPT_mllvm))
177 Add("-mllvm:" + StringRef(A->getValue()));
178
Martin Storsjo107b5482017-11-03 07:18:37179 for (auto *A : Args.filtered(OPT_Xlink))
180 Add(A->getValue());
181
Rui Ueyama5941b0b2017-09-11 20:14:47182 if (Args.getLastArgValue(OPT_m) == "i386pe")
183 Add("-alternatename:__image_base__=___ImageBase");
184 else
185 Add("-alternatename:__image_base__=__ImageBase");
186
187 std::vector<StringRef> SearchPaths;
188 for (auto *A : Args.filtered(OPT_L))
189 SearchPaths.push_back(A->getValue());
190
Martin Storsjo32e16262017-09-13 07:28:09191 StringRef Prefix = "";
Martin Storsjo2ba37d42017-09-13 19:29:44192 bool Static = false;
193 for (auto *A : Args) {
Rui Ueyama9c37d8c2017-09-13 19:44:36194 switch (A->getOption().getUnaliasedOption().getID()) {
Martin Storsjo32e16262017-09-13 07:28:09195 case OPT_INPUT:
Rui Ueyamab3592632017-09-14 18:33:09196 if (StringRef(A->getValue()).endswith_lower(".def"))
Martin Storsjo064b0fa2017-09-13 07:28:13197 Add("-def:" + StringRef(A->getValue()));
198 else
199 Add(Prefix + StringRef(A->getValue()));
Martin Storsjo32e16262017-09-13 07:28:09200 break;
201 case OPT_l:
Martin Storsjo2ba37d42017-09-13 19:29:44202 Add(Prefix + searchLibrary(A->getValue(), SearchPaths, Static));
Martin Storsjo32e16262017-09-13 07:28:09203 break;
204 case OPT_whole_archive:
205 Prefix = "-wholearchive:";
206 break;
207 case OPT_no_whole_archive:
208 Prefix = "";
209 break;
Martin Storsjo2ba37d42017-09-13 19:29:44210 case OPT_Bstatic:
211 Static = true;
212 break;
213 case OPT_Bdynamic:
214 Static = false;
215 break;
Martin Storsjo32e16262017-09-13 07:28:09216 }
Rui Ueyama5941b0b2017-09-11 20:14:47217 }
218
Martin Storsjoa79762a2017-09-11 20:43:39219 if (Args.hasArg(OPT_verbose) || Args.hasArg(OPT__HASH_HASH_HASH))
220 outs() << llvm::join(LinkArgs, " ") << "\n";
221
222 if (Args.hasArg(OPT__HASH_HASH_HASH))
223 return true;
224
Rui Ueyama5941b0b2017-09-11 20:14:47225 // Repack vector of strings to vector of const char pointers for coff::link.
Martell Malone894dbbe2017-09-11 17:02:59226 std::vector<const char *> Vec;
227 for (const std::string &S : LinkArgs)
228 Vec.push_back(S.c_str());
Rui Ueyama6f4e2552017-10-23 20:03:32229 return coff::link(Vec, true);
Martell Malone894dbbe2017-09-11 17:02:59230}