blob: dc9b046c5d13c579b58e993eefae75b451cd5c7f [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"
Rui Ueyama338c4b22017-12-12 20:34:3815#include "lld/Common/ErrorHandler.h"
Martell Malone894dbbe2017-09-11 17:02:5916#include "llvm/ADT/ArrayRef.h"
17#include "llvm/ADT/Optional.h"
Martin Storsjoa79762a2017-09-11 20:43:3918#include "llvm/ADT/StringExtras.h"
Martell Malone894dbbe2017-09-11 17:02:5919#include "llvm/ADT/StringRef.h"
Martell Malone894dbbe2017-09-11 17:02:5920#include "llvm/Option/Arg.h"
21#include "llvm/Option/ArgList.h"
22#include "llvm/Option/Option.h"
23#include "llvm/Support/CommandLine.h"
24#include "llvm/Support/FileSystem.h"
25#include "llvm/Support/Path.h"
26
27#if !defined(_MSC_VER) && !defined(__MINGW32__)
28#include <unistd.h>
29#endif
30
31using namespace lld;
32using namespace llvm;
33
Martell Malone894dbbe2017-09-11 17:02:5934// Create OptTable
35enum {
36 OPT_INVALID = 0,
37#define OPTION(_1, _2, ID, _4, _5, _6, _7, _8, _9, _10, _11, _12) OPT_##ID,
38#include "Options.inc"
39#undef OPTION
40};
41
42// Create prefix string literals used in Options.td
Rui Ueyamac16fd252017-09-11 22:04:1343#define PREFIX(NAME, VALUE) static const char *const NAME[] = VALUE;
Martell Malone894dbbe2017-09-11 17:02:5944#include "Options.inc"
45#undef PREFIX
46
47// Create table mapping all options defined in Options.td
48static const opt::OptTable::Info InfoTable[] = {
49#define OPTION(X1, X2, ID, KIND, GROUP, ALIAS, X7, X8, X9, X10, X11, X12) \
50 {X1, X2, X10, X11, OPT_##ID, opt::Option::KIND##Class, \
51 X9, X8, OPT_##GROUP, OPT_##ALIAS, X7, X12},
52#include "Options.inc"
53#undef OPTION
54};
55
Rui Ueyamac16fd252017-09-11 22:04:1356namespace {
Rui Ueyama44747962017-09-11 22:04:2557class MinGWOptTable : public opt::OptTable {
Martell Malone894dbbe2017-09-11 17:02:5958public:
Rui Ueyama44747962017-09-11 22:04:2559 MinGWOptTable() : OptTable(InfoTable, false) {}
Martell Malone894dbbe2017-09-11 17:02:5960 opt::InputArgList parse(ArrayRef<const char *> Argv);
61};
Martell Malone894dbbe2017-09-11 17:02:5962} // namespace
63
Rui Ueyamad2da71f2017-09-11 22:04:3764opt::InputArgList MinGWOptTable::parse(ArrayRef<const char *> Argv) {
65 unsigned MissingIndex;
66 unsigned MissingCount;
67
68 SmallVector<const char *, 256> Vec(Argv.data(), Argv.data() + Argv.size());
69 opt::InputArgList Args = this->ParseArgs(Vec, MissingIndex, MissingCount);
70
71 if (MissingCount)
Rui Ueyama338c4b22017-12-12 20:34:3872 fatal(StringRef(Args.getArgString(MissingIndex)) + ": missing argument");
Rui Ueyamad2da71f2017-09-11 22:04:3773 for (auto *Arg : Args.filtered(OPT_UNKNOWN))
Rui Ueyama338c4b22017-12-12 20:34:3874 fatal("unknown argument: " + Arg->getSpelling());
Rui Ueyamaa835bab2017-09-13 20:30:5975 if (!Args.hasArg(OPT_INPUT) && !Args.hasArg(OPT_l))
Rui Ueyama338c4b22017-12-12 20:34:3876 fatal("no input files");
Rui Ueyamad2da71f2017-09-11 22:04:3777 return Args;
Martell Malone894dbbe2017-09-11 17:02:5978}
79
80// Find a file by concatenating given paths.
81static Optional<std::string> findFile(StringRef Path1, const Twine &Path2) {
82 SmallString<128> S;
83 sys::path::append(S, Path1, Path2);
84 if (sys::fs::exists(S))
85 return S.str().str();
86 return None;
87}
88
Martell Malone894dbbe2017-09-11 17:02:5989// This is for -lfoo. We'll look for libfoo.dll.a or libfoo.a from search paths.
Rui Ueyama5941b0b2017-09-11 20:14:4790static std::string
91searchLibrary(StringRef Name, ArrayRef<StringRef> SearchPaths, bool BStatic) {
92 if (Name.startswith(":")) {
93 for (StringRef Dir : SearchPaths)
94 if (Optional<std::string> S = findFile(Dir, Name.substr(1)))
95 return *S;
Rui Ueyama338c4b22017-12-12 20:34:3896 fatal("unable to find library -l" + Name);
Martell Malone894dbbe2017-09-11 17:02:5997 }
Martell Malone894dbbe2017-09-11 17:02:5998
Rui Ueyama5941b0b2017-09-11 20:14:4799 for (StringRef Dir : SearchPaths) {
100 if (!BStatic)
101 if (Optional<std::string> S = findFile(Dir, "lib" + Name + ".dll.a"))
102 return *S;
103 if (Optional<std::string> S = findFile(Dir, "lib" + Name + ".a"))
104 return *S;
Martell Malone894dbbe2017-09-11 17:02:59105 }
Rui Ueyama338c4b22017-12-12 20:34:38106 fatal("unable to find library -l" + Name);
Martell Malone894dbbe2017-09-11 17:02:59107}
108
Rui Ueyama5941b0b2017-09-11 20:14:47109// Convert Unix-ish command line arguments to Windows-ish ones and
110// then call coff::link.
Rui Ueyama95114b42017-09-11 22:04:01111bool mingw::link(ArrayRef<const char *> ArgsArr, raw_ostream &Diag) {
Rui Ueyama44747962017-09-11 22:04:25112 MinGWOptTable Parser;
Martell Malone894dbbe2017-09-11 17:02:59113 opt::InputArgList Args = Parser.parse(ArgsArr.slice(1));
Martell Malone894dbbe2017-09-11 17:02:59114
Rui Ueyama5941b0b2017-09-11 20:14:47115 std::vector<std::string> LinkArgs;
116 auto Add = [&](const Twine &S) { LinkArgs.push_back(S.str()); };
Martell Malone894dbbe2017-09-11 17:02:59117
Martin Storsjoa79762a2017-09-11 20:43:39118 Add("lld-link");
Martin Storsjo31fe4cd2017-09-13 19:29:39119 Add("-lldmingw");
Martell Malone894dbbe2017-09-11 17:02:59120
Martin Storsjo6d8dace2017-09-12 19:23:54121 if (auto *A = Args.getLastArg(OPT_entry)) {
122 StringRef S = A->getValue();
123 if (Args.getLastArgValue(OPT_m) == "i386pe" && S.startswith("_"))
124 Add("-entry:" + S.substr(1));
125 else
126 Add("-entry:" + S);
127 }
128
Rui Ueyama5941b0b2017-09-11 20:14:47129 if (auto *A = Args.getLastArg(OPT_subs))
130 Add("-subsystem:" + StringRef(A->getValue()));
Rui Ueyamac3e46022017-09-11 20:54:51131 if (auto *A = Args.getLastArg(OPT_out_implib))
Rui Ueyama5941b0b2017-09-11 20:14:47132 Add("-implib:" + StringRef(A->getValue()));
133 if (auto *A = Args.getLastArg(OPT_stack))
134 Add("-stack:" + StringRef(A->getValue()));
Martin Storsjoa84a4772017-10-12 05:37:18135 if (auto *A = Args.getLastArg(OPT_output_def))
136 Add("-output-def:" + StringRef(A->getValue()));
Martin Storsjoe9168682017-11-15 08:18:15137 if (auto *A = Args.getLastArg(OPT_image_base))
138 Add("-base:" + StringRef(A->getValue()));
Martin Storsjo03661552018-05-15 21:12:29139 if (auto *A = Args.getLastArg(OPT_map))
140 Add("-lldmap:" + 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
Martin Storsjob7d50112018-05-15 06:34:18149 if (auto *A = Args.getLastArg(OPT_pdb)) {
150 Add("-debug");
151 Add("-pdb:" + StringRef(A->getValue()));
Martin Storsjoc0826742018-06-29 06:08:31152 } else if (Args.hasArg(OPT_strip_debug)) {
153 Add("-debug:symtab");
Martin Storsjob7d50112018-05-15 06:34:18154 } else if (!Args.hasArg(OPT_strip_all)) {
155 Add("-debug:dwarf");
156 }
157
Rui Ueyama5941b0b2017-09-11 20:14:47158 if (Args.hasArg(OPT_shared))
159 Add("-dll");
Rui Ueyama7d306a52017-09-13 19:40:07160 if (Args.hasArg(OPT_verbose))
161 Add("-verbose");
Martin Storsjoa84a4772017-10-12 05:37:18162 if (Args.hasArg(OPT_export_all_symbols))
163 Add("-export-all-symbols");
Martin Storsjo3c8d4b52017-11-15 08:18:06164 if (Args.hasArg(OPT_large_address_aware))
165 Add("-largeaddressaware");
Martin Storsjofb36e6e2018-03-14 20:17:24166 if (Args.hasArg(OPT_kill_at))
167 Add("-kill-at");
Rui Ueyama5941b0b2017-09-11 20:14:47168
Martin Storsjo71663a22017-12-12 19:39:18169 if (Args.getLastArgValue(OPT_m) != "thumb2pe" &&
170 Args.getLastArgValue(OPT_m) != "arm64pe" && !Args.hasArg(OPT_dynamicbase))
171 Add("-dynamicbase:no");
Martin Storsjo094d8c02017-11-15 08:18:11172
Martin Storsjob190fd22017-11-15 08:18:20173 if (Args.hasFlag(OPT_gc_sections, OPT_no_gc_sections, false))
174 Add("-opt:ref");
175 else
176 Add("-opt:noref");
177
178 if (auto *A = Args.getLastArg(OPT_icf)) {
179 StringRef S = A->getValue();
180 if (S == "all")
181 Add("-opt:icf");
182 else if (S == "safe" || S == "none")
183 Add("-opt:noicf");
184 else
Rui Ueyama338c4b22017-12-12 20:34:38185 fatal("unknown parameter: --icf=" + S);
Martin Storsjob190fd22017-11-15 08:18:20186 } else {
187 Add("-opt:noicf");
188 }
189
Rui Ueyama5941b0b2017-09-11 20:14:47190 if (auto *A = Args.getLastArg(OPT_m)) {
191 StringRef S = A->getValue();
192 if (S == "i386pe")
193 Add("-machine:x86");
194 else if (S == "i386pep")
195 Add("-machine:x64");
196 else if (S == "thumb2pe")
197 Add("-machine:arm");
198 else if (S == "arm64pe")
199 Add("-machine:arm64");
200 else
Rui Ueyama338c4b22017-12-12 20:34:38201 fatal("unknown parameter: -m" + S);
Rui Ueyama5941b0b2017-09-11 20:14:47202 }
203
Martell Malone0d176382017-09-11 21:36:37204 for (auto *A : Args.filtered(OPT_mllvm))
205 Add("-mllvm:" + StringRef(A->getValue()));
206
Martin Storsjo107b5482017-11-03 07:18:37207 for (auto *A : Args.filtered(OPT_Xlink))
208 Add(A->getValue());
209
Rui Ueyama5941b0b2017-09-11 20:14:47210 if (Args.getLastArgValue(OPT_m) == "i386pe")
211 Add("-alternatename:__image_base__=___ImageBase");
212 else
213 Add("-alternatename:__image_base__=__ImageBase");
214
Martin Storsjodb629132018-09-10 17:41:40215 for (auto *A : Args.filtered(OPT_require_defined))
216 Add("-include:" + StringRef(A->getValue()));
217
Rui Ueyama5941b0b2017-09-11 20:14:47218 std::vector<StringRef> SearchPaths;
219 for (auto *A : Args.filtered(OPT_L))
220 SearchPaths.push_back(A->getValue());
221
Martin Storsjo32e16262017-09-13 07:28:09222 StringRef Prefix = "";
Martin Storsjo2ba37d42017-09-13 19:29:44223 bool Static = false;
224 for (auto *A : Args) {
Rui Ueyama9c37d8c2017-09-13 19:44:36225 switch (A->getOption().getUnaliasedOption().getID()) {
Martin Storsjo32e16262017-09-13 07:28:09226 case OPT_INPUT:
Rui Ueyamab3592632017-09-14 18:33:09227 if (StringRef(A->getValue()).endswith_lower(".def"))
Martin Storsjo064b0fa2017-09-13 07:28:13228 Add("-def:" + StringRef(A->getValue()));
229 else
230 Add(Prefix + StringRef(A->getValue()));
Martin Storsjo32e16262017-09-13 07:28:09231 break;
232 case OPT_l:
Martin Storsjo2ba37d42017-09-13 19:29:44233 Add(Prefix + searchLibrary(A->getValue(), SearchPaths, Static));
Martin Storsjo32e16262017-09-13 07:28:09234 break;
235 case OPT_whole_archive:
236 Prefix = "-wholearchive:";
237 break;
238 case OPT_no_whole_archive:
239 Prefix = "";
240 break;
Martin Storsjo2ba37d42017-09-13 19:29:44241 case OPT_Bstatic:
242 Static = true;
243 break;
244 case OPT_Bdynamic:
245 Static = false;
246 break;
Martin Storsjo32e16262017-09-13 07:28:09247 }
Rui Ueyama5941b0b2017-09-11 20:14:47248 }
249
Martin Storsjoa79762a2017-09-11 20:43:39250 if (Args.hasArg(OPT_verbose) || Args.hasArg(OPT__HASH_HASH_HASH))
251 outs() << llvm::join(LinkArgs, " ") << "\n";
252
253 if (Args.hasArg(OPT__HASH_HASH_HASH))
254 return true;
255
Rui Ueyama5941b0b2017-09-11 20:14:47256 // Repack vector of strings to vector of const char pointers for coff::link.
Martell Malone894dbbe2017-09-11 17:02:59257 std::vector<const char *> Vec;
258 for (const std::string &S : LinkArgs)
259 Vec.push_back(S.c_str());
Rui Ueyama6f4e2552017-10-23 20:03:32260 return coff::link(Vec, true);
Martell Malone894dbbe2017-09-11 17:02:59261}