1ece8a530Spatrick //===- MinGW/Driver.cpp ---------------------------------------------------===//
2ece8a530Spatrick //
3ece8a530Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4ece8a530Spatrick // See https://llvm.org/LICENSE.txt for license information.
5ece8a530Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6ece8a530Spatrick //
7ece8a530Spatrick //===----------------------------------------------------------------------===//
8ece8a530Spatrick //
9ece8a530Spatrick // MinGW is a GNU development environment for Windows. It consists of GNU
10ece8a530Spatrick // tools such as GCC and GNU ld. Unlike Cygwin, there's no POSIX-compatible
11ece8a530Spatrick // layer, as it aims to be a native development toolchain.
12ece8a530Spatrick //
13ece8a530Spatrick // lld/MinGW is a drop-in replacement for GNU ld/MinGW.
14ece8a530Spatrick //
15ece8a530Spatrick // Being a native development tool, a MinGW linker is not very different from
16ece8a530Spatrick // Microsoft link.exe, so a MinGW linker can be implemented as a thin wrapper
17ece8a530Spatrick // for lld/COFF. This driver takes Unix-ish command line options, translates
18ece8a530Spatrick // them to Windows-ish ones, and then passes them to lld/COFF.
19ece8a530Spatrick //
20ece8a530Spatrick // When this driver calls the lld/COFF driver, it passes a hidden option
21ece8a530Spatrick // "-lldmingw" along with other user-supplied options, to run the lld/COFF
22ece8a530Spatrick // linker in "MinGW mode".
23ece8a530Spatrick //
24ece8a530Spatrick // There are subtle differences between MS link.exe and GNU ld/MinGW, and GNU
25ece8a530Spatrick // ld/MinGW implements a few GNU-specific features. Such features are directly
26ece8a530Spatrick // implemented in lld/COFF and enabled only when the linker is running in MinGW
27ece8a530Spatrick // mode.
28ece8a530Spatrick //
29ece8a530Spatrick //===----------------------------------------------------------------------===//
30ece8a530Spatrick
31ece8a530Spatrick #include "lld/Common/Driver.h"
32*dfe94b16Srobert #include "lld/Common/CommonLinkerContext.h"
33ece8a530Spatrick #include "lld/Common/ErrorHandler.h"
34ece8a530Spatrick #include "lld/Common/Memory.h"
35ece8a530Spatrick #include "lld/Common/Version.h"
36ece8a530Spatrick #include "llvm/ADT/ArrayRef.h"
37ece8a530Spatrick #include "llvm/ADT/StringExtras.h"
38ece8a530Spatrick #include "llvm/ADT/StringRef.h"
39ece8a530Spatrick #include "llvm/ADT/Triple.h"
40ece8a530Spatrick #include "llvm/Option/Arg.h"
41ece8a530Spatrick #include "llvm/Option/ArgList.h"
42ece8a530Spatrick #include "llvm/Option/Option.h"
43ece8a530Spatrick #include "llvm/Support/CommandLine.h"
44ece8a530Spatrick #include "llvm/Support/FileSystem.h"
45bb684c34Spatrick #include "llvm/Support/Host.h"
46ece8a530Spatrick #include "llvm/Support/Path.h"
47*dfe94b16Srobert #include <optional>
48ece8a530Spatrick
49ece8a530Spatrick #if !defined(_MSC_VER) && !defined(__MINGW32__)
50ece8a530Spatrick #include <unistd.h>
51ece8a530Spatrick #endif
52ece8a530Spatrick
53ece8a530Spatrick using namespace lld;
54ece8a530Spatrick using namespace llvm;
55ece8a530Spatrick
56ece8a530Spatrick // Create OptTable
57ece8a530Spatrick enum {
58ece8a530Spatrick OPT_INVALID = 0,
59ece8a530Spatrick #define OPTION(_1, _2, ID, _4, _5, _6, _7, _8, _9, _10, _11, _12) OPT_##ID,
60ece8a530Spatrick #include "Options.inc"
61ece8a530Spatrick #undef OPTION
62ece8a530Spatrick };
63ece8a530Spatrick
64ece8a530Spatrick // Create prefix string literals used in Options.td
65*dfe94b16Srobert #define PREFIX(NAME, VALUE) \
66*dfe94b16Srobert static constexpr llvm::StringLiteral NAME##_init[] = VALUE; \
67*dfe94b16Srobert static constexpr llvm::ArrayRef<llvm::StringLiteral> NAME( \
68*dfe94b16Srobert NAME##_init, std::size(NAME##_init) - 1);
69ece8a530Spatrick #include "Options.inc"
70ece8a530Spatrick #undef PREFIX
71ece8a530Spatrick
72ece8a530Spatrick // Create table mapping all options defined in Options.td
73*dfe94b16Srobert static constexpr opt::OptTable::Info infoTable[] = {
74ece8a530Spatrick #define OPTION(X1, X2, ID, KIND, GROUP, ALIAS, X7, X8, X9, X10, X11, X12) \
75ece8a530Spatrick {X1, X2, X10, X11, OPT_##ID, opt::Option::KIND##Class, \
76ece8a530Spatrick X9, X8, OPT_##GROUP, OPT_##ALIAS, X7, X12},
77ece8a530Spatrick #include "Options.inc"
78ece8a530Spatrick #undef OPTION
79ece8a530Spatrick };
80ece8a530Spatrick
81ece8a530Spatrick namespace {
82*dfe94b16Srobert class MinGWOptTable : public opt::GenericOptTable {
83ece8a530Spatrick public:
MinGWOptTable()84*dfe94b16Srobert MinGWOptTable() : opt::GenericOptTable(infoTable, false) {}
85ece8a530Spatrick opt::InputArgList parse(ArrayRef<const char *> argv);
86ece8a530Spatrick };
87ece8a530Spatrick } // namespace
88ece8a530Spatrick
printHelp(const char * argv0)89ece8a530Spatrick static void printHelp(const char *argv0) {
901cf9926bSpatrick MinGWOptTable().printHelp(
91ece8a530Spatrick lld::outs(), (std::string(argv0) + " [options] file...").c_str(), "lld",
92ece8a530Spatrick false /*ShowHidden*/, true /*ShowAllAliases*/);
93ece8a530Spatrick lld::outs() << "\n";
94ece8a530Spatrick }
95ece8a530Spatrick
getQuotingStyle()96ece8a530Spatrick static cl::TokenizerCallback getQuotingStyle() {
97ece8a530Spatrick if (Triple(sys::getProcessTriple()).getOS() == Triple::Win32)
98ece8a530Spatrick return cl::TokenizeWindowsCommandLine;
99ece8a530Spatrick return cl::TokenizeGNUCommandLine;
100ece8a530Spatrick }
101ece8a530Spatrick
parse(ArrayRef<const char * > argv)102ece8a530Spatrick opt::InputArgList MinGWOptTable::parse(ArrayRef<const char *> argv) {
103ece8a530Spatrick unsigned missingIndex;
104ece8a530Spatrick unsigned missingCount;
105ece8a530Spatrick
106ece8a530Spatrick SmallVector<const char *, 256> vec(argv.data(), argv.data() + argv.size());
107*dfe94b16Srobert cl::ExpandResponseFiles(saver(), getQuotingStyle(), vec);
108ece8a530Spatrick opt::InputArgList args = this->ParseArgs(vec, missingIndex, missingCount);
109ece8a530Spatrick
110ece8a530Spatrick if (missingCount)
111ece8a530Spatrick error(StringRef(args.getArgString(missingIndex)) + ": missing argument");
112ece8a530Spatrick for (auto *arg : args.filtered(OPT_UNKNOWN))
113ece8a530Spatrick error("unknown argument: " + arg->getAsString(args));
114ece8a530Spatrick return args;
115ece8a530Spatrick }
116ece8a530Spatrick
117ece8a530Spatrick // Find a file by concatenating given paths.
findFile(StringRef path1,const Twine & path2)118*dfe94b16Srobert static std::optional<std::string> findFile(StringRef path1,
119*dfe94b16Srobert const Twine &path2) {
120ece8a530Spatrick SmallString<128> s;
121ece8a530Spatrick sys::path::append(s, path1, path2);
122ece8a530Spatrick if (sys::fs::exists(s))
123bb684c34Spatrick return std::string(s);
124*dfe94b16Srobert return std::nullopt;
125ece8a530Spatrick }
126ece8a530Spatrick
127ece8a530Spatrick // This is for -lfoo. We'll look for libfoo.dll.a or libfoo.a from search paths.
128ece8a530Spatrick static std::string
searchLibrary(StringRef name,ArrayRef<StringRef> searchPaths,bool bStatic)129ece8a530Spatrick searchLibrary(StringRef name, ArrayRef<StringRef> searchPaths, bool bStatic) {
130ece8a530Spatrick if (name.startswith(":")) {
131ece8a530Spatrick for (StringRef dir : searchPaths)
132*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, name.substr(1)))
133ece8a530Spatrick return *s;
134ece8a530Spatrick error("unable to find library -l" + name);
135ece8a530Spatrick return "";
136ece8a530Spatrick }
137ece8a530Spatrick
138ece8a530Spatrick for (StringRef dir : searchPaths) {
139ece8a530Spatrick if (!bStatic) {
140*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, "lib" + name + ".dll.a"))
141ece8a530Spatrick return *s;
142*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, name + ".dll.a"))
143ece8a530Spatrick return *s;
144ece8a530Spatrick }
145*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, "lib" + name + ".a"))
146*dfe94b16Srobert return *s;
147*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, name + ".lib"))
148ece8a530Spatrick return *s;
149ece8a530Spatrick if (!bStatic) {
150*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, "lib" + name + ".dll"))
151ece8a530Spatrick return *s;
152*dfe94b16Srobert if (std::optional<std::string> s = findFile(dir, name + ".dll"))
1531cf9926bSpatrick return *s;
154ece8a530Spatrick }
155ece8a530Spatrick }
156ece8a530Spatrick error("unable to find library -l" + name);
157ece8a530Spatrick return "";
158ece8a530Spatrick }
159ece8a530Spatrick
160ece8a530Spatrick // Convert Unix-ish command line arguments to Windows-ish ones and
161ece8a530Spatrick // then call coff::link.
link(ArrayRef<const char * > argsArr,llvm::raw_ostream & stdoutOS,llvm::raw_ostream & stderrOS,bool exitEarly,bool disableOutput)162*dfe94b16Srobert bool mingw::link(ArrayRef<const char *> argsArr, llvm::raw_ostream &stdoutOS,
163*dfe94b16Srobert llvm::raw_ostream &stderrOS, bool exitEarly,
164*dfe94b16Srobert bool disableOutput) {
165*dfe94b16Srobert auto *ctx = new CommonLinkerContext;
166*dfe94b16Srobert ctx->e.initialize(stdoutOS, stderrOS, exitEarly, disableOutput);
167ece8a530Spatrick
168ece8a530Spatrick MinGWOptTable parser;
169ece8a530Spatrick opt::InputArgList args = parser.parse(argsArr.slice(1));
170ece8a530Spatrick
171ece8a530Spatrick if (errorCount())
172ece8a530Spatrick return false;
173ece8a530Spatrick
174ece8a530Spatrick if (args.hasArg(OPT_help)) {
175ece8a530Spatrick printHelp(argsArr[0]);
176ece8a530Spatrick return true;
177ece8a530Spatrick }
178ece8a530Spatrick
179ece8a530Spatrick // A note about "compatible with GNU linkers" message: this is a hack for
180ece8a530Spatrick // scripts generated by GNU Libtool 2.4.6 (released in February 2014 and
181ece8a530Spatrick // still the newest version in March 2017) or earlier to recognize LLD as
182ece8a530Spatrick // a GNU compatible linker. As long as an output for the -v option
183ece8a530Spatrick // contains "GNU" or "with BFD", they recognize us as GNU-compatible.
184ece8a530Spatrick if (args.hasArg(OPT_v) || args.hasArg(OPT_version))
185ece8a530Spatrick message(getLLDVersion() + " (compatible with GNU linkers)");
186ece8a530Spatrick
187ece8a530Spatrick // The behavior of -v or --version is a bit strange, but this is
188ece8a530Spatrick // needed for compatibility with GNU linkers.
189ece8a530Spatrick if (args.hasArg(OPT_v) && !args.hasArg(OPT_INPUT) && !args.hasArg(OPT_l))
190ece8a530Spatrick return true;
191ece8a530Spatrick if (args.hasArg(OPT_version))
192ece8a530Spatrick return true;
193ece8a530Spatrick
194ece8a530Spatrick if (!args.hasArg(OPT_INPUT) && !args.hasArg(OPT_l)) {
195ece8a530Spatrick error("no input files");
196ece8a530Spatrick return false;
197ece8a530Spatrick }
198ece8a530Spatrick
199ece8a530Spatrick std::vector<std::string> linkArgs;
200ece8a530Spatrick auto add = [&](const Twine &s) { linkArgs.push_back(s.str()); };
201ece8a530Spatrick
202ece8a530Spatrick add("lld-link");
203ece8a530Spatrick add("-lldmingw");
204ece8a530Spatrick
205ece8a530Spatrick if (auto *a = args.getLastArg(OPT_entry)) {
206ece8a530Spatrick StringRef s = a->getValue();
207ece8a530Spatrick if (args.getLastArgValue(OPT_m) == "i386pe" && s.startswith("_"))
208ece8a530Spatrick add("-entry:" + s.substr(1));
209ece8a530Spatrick else
210ece8a530Spatrick add("-entry:" + s);
211ece8a530Spatrick }
212ece8a530Spatrick
213ece8a530Spatrick if (args.hasArg(OPT_major_os_version, OPT_minor_os_version,
214ece8a530Spatrick OPT_major_subsystem_version, OPT_minor_subsystem_version)) {
2151cf9926bSpatrick StringRef majOSVer = args.getLastArgValue(OPT_major_os_version, "6");
2161cf9926bSpatrick StringRef minOSVer = args.getLastArgValue(OPT_minor_os_version, "0");
2171cf9926bSpatrick StringRef majSubSysVer = "6";
2181cf9926bSpatrick StringRef minSubSysVer = "0";
2191cf9926bSpatrick StringRef subSysName = "default";
2201cf9926bSpatrick StringRef subSysVer;
2211cf9926bSpatrick // Iterate over --{major,minor}-subsystem-version and --subsystem, and pick
2221cf9926bSpatrick // the version number components from the last one of them that specifies
2231cf9926bSpatrick // a version.
2241cf9926bSpatrick for (auto *a : args.filtered(OPT_major_subsystem_version,
2251cf9926bSpatrick OPT_minor_subsystem_version, OPT_subs)) {
2261cf9926bSpatrick switch (a->getOption().getID()) {
2271cf9926bSpatrick case OPT_major_subsystem_version:
2281cf9926bSpatrick majSubSysVer = a->getValue();
2291cf9926bSpatrick break;
2301cf9926bSpatrick case OPT_minor_subsystem_version:
2311cf9926bSpatrick minSubSysVer = a->getValue();
2321cf9926bSpatrick break;
2331cf9926bSpatrick case OPT_subs:
2341cf9926bSpatrick std::tie(subSysName, subSysVer) = StringRef(a->getValue()).split(':');
2351cf9926bSpatrick if (!subSysVer.empty()) {
2361cf9926bSpatrick if (subSysVer.contains('.'))
2371cf9926bSpatrick std::tie(majSubSysVer, minSubSysVer) = subSysVer.split('.');
2381cf9926bSpatrick else
2391cf9926bSpatrick majSubSysVer = subSysVer;
2401cf9926bSpatrick }
2411cf9926bSpatrick break;
2421cf9926bSpatrick }
2431cf9926bSpatrick }
2441cf9926bSpatrick add("-osversion:" + majOSVer + "." + minOSVer);
2451cf9926bSpatrick add("-subsystem:" + subSysName + "," + majSubSysVer + "." + minSubSysVer);
2461cf9926bSpatrick } else if (args.hasArg(OPT_subs)) {
247ece8a530Spatrick StringRef subSys = args.getLastArgValue(OPT_subs, "default");
2481cf9926bSpatrick StringRef subSysName, subSysVer;
2491cf9926bSpatrick std::tie(subSysName, subSysVer) = subSys.split(':');
2501cf9926bSpatrick StringRef sep = subSysVer.empty() ? "" : ",";
2511cf9926bSpatrick add("-subsystem:" + subSysName + sep + subSysVer);
252ece8a530Spatrick }
253ece8a530Spatrick
254ece8a530Spatrick if (auto *a = args.getLastArg(OPT_out_implib))
255ece8a530Spatrick add("-implib:" + StringRef(a->getValue()));
256ece8a530Spatrick if (auto *a = args.getLastArg(OPT_stack))
257ece8a530Spatrick add("-stack:" + StringRef(a->getValue()));
258ece8a530Spatrick if (auto *a = args.getLastArg(OPT_output_def))
259ece8a530Spatrick add("-output-def:" + StringRef(a->getValue()));
260ece8a530Spatrick if (auto *a = args.getLastArg(OPT_image_base))
261ece8a530Spatrick add("-base:" + StringRef(a->getValue()));
262ece8a530Spatrick if (auto *a = args.getLastArg(OPT_map))
263ece8a530Spatrick add("-lldmap:" + StringRef(a->getValue()));
264ece8a530Spatrick if (auto *a = args.getLastArg(OPT_reproduce))
265ece8a530Spatrick add("-reproduce:" + StringRef(a->getValue()));
266bb684c34Spatrick if (auto *a = args.getLastArg(OPT_thinlto_cache_dir))
267bb684c34Spatrick add("-lldltocache:" + StringRef(a->getValue()));
268bb684c34Spatrick if (auto *a = args.getLastArg(OPT_file_alignment))
269bb684c34Spatrick add("-filealign:" + StringRef(a->getValue()));
270bb684c34Spatrick if (auto *a = args.getLastArg(OPT_section_alignment))
271bb684c34Spatrick add("-align:" + StringRef(a->getValue()));
272*dfe94b16Srobert if (auto *a = args.getLastArg(OPT_heap))
273*dfe94b16Srobert add("-heap:" + StringRef(a->getValue()));
274ece8a530Spatrick
275ece8a530Spatrick if (auto *a = args.getLastArg(OPT_o))
276ece8a530Spatrick add("-out:" + StringRef(a->getValue()));
277ece8a530Spatrick else if (args.hasArg(OPT_shared))
278ece8a530Spatrick add("-out:a.dll");
279ece8a530Spatrick else
280ece8a530Spatrick add("-out:a.exe");
281ece8a530Spatrick
282ece8a530Spatrick if (auto *a = args.getLastArg(OPT_pdb)) {
283ece8a530Spatrick add("-debug");
284ece8a530Spatrick StringRef v = a->getValue();
285ece8a530Spatrick if (!v.empty())
286ece8a530Spatrick add("-pdb:" + v);
287ece8a530Spatrick } else if (args.hasArg(OPT_strip_debug)) {
288ece8a530Spatrick add("-debug:symtab");
289ece8a530Spatrick } else if (!args.hasArg(OPT_strip_all)) {
290ece8a530Spatrick add("-debug:dwarf");
291ece8a530Spatrick }
292ece8a530Spatrick
2931cf9926bSpatrick if (args.hasFlag(OPT_fatal_warnings, OPT_no_fatal_warnings, false))
2941cf9926bSpatrick add("-WX");
2951cf9926bSpatrick else
2961cf9926bSpatrick add("-WX:no");
2971cf9926bSpatrick
2981cf9926bSpatrick if (args.hasFlag(OPT_enable_stdcall_fixup, OPT_disable_stdcall_fixup, false))
2991cf9926bSpatrick add("-stdcall-fixup");
3001cf9926bSpatrick else if (args.hasArg(OPT_disable_stdcall_fixup))
3011cf9926bSpatrick add("-stdcall-fixup:no");
3021cf9926bSpatrick
303ece8a530Spatrick if (args.hasArg(OPT_shared))
304ece8a530Spatrick add("-dll");
305ece8a530Spatrick if (args.hasArg(OPT_verbose))
306ece8a530Spatrick add("-verbose");
307ece8a530Spatrick if (args.hasArg(OPT_exclude_all_symbols))
308ece8a530Spatrick add("-exclude-all-symbols");
309ece8a530Spatrick if (args.hasArg(OPT_export_all_symbols))
310ece8a530Spatrick add("-export-all-symbols");
311ece8a530Spatrick if (args.hasArg(OPT_large_address_aware))
312ece8a530Spatrick add("-largeaddressaware");
313ece8a530Spatrick if (args.hasArg(OPT_kill_at))
314ece8a530Spatrick add("-kill-at");
315ece8a530Spatrick if (args.hasArg(OPT_appcontainer))
316ece8a530Spatrick add("-appcontainer");
3171cf9926bSpatrick if (args.hasFlag(OPT_no_seh, OPT_disable_no_seh, false))
318bb684c34Spatrick add("-noseh");
319ece8a530Spatrick
320ece8a530Spatrick if (args.getLastArgValue(OPT_m) != "thumb2pe" &&
3211cf9926bSpatrick args.getLastArgValue(OPT_m) != "arm64pe" &&
3221cf9926bSpatrick args.hasFlag(OPT_disable_dynamicbase, OPT_dynamicbase, false))
323ece8a530Spatrick add("-dynamicbase:no");
3241cf9926bSpatrick if (args.hasFlag(OPT_disable_high_entropy_va, OPT_high_entropy_va, false))
3251cf9926bSpatrick add("-highentropyva:no");
3261cf9926bSpatrick if (args.hasFlag(OPT_disable_nxcompat, OPT_nxcompat, false))
3271cf9926bSpatrick add("-nxcompat:no");
3281cf9926bSpatrick if (args.hasFlag(OPT_disable_tsaware, OPT_tsaware, false))
3291cf9926bSpatrick add("-tsaware:no");
330ece8a530Spatrick
331*dfe94b16Srobert if (args.hasFlag(OPT_disable_reloc_section, OPT_enable_reloc_section, false))
332*dfe94b16Srobert add("-fixed");
333*dfe94b16Srobert
334ece8a530Spatrick if (args.hasFlag(OPT_no_insert_timestamp, OPT_insert_timestamp, false))
335ece8a530Spatrick add("-timestamp:0");
336ece8a530Spatrick
337ece8a530Spatrick if (args.hasFlag(OPT_gc_sections, OPT_no_gc_sections, false))
338ece8a530Spatrick add("-opt:ref");
339ece8a530Spatrick else
340ece8a530Spatrick add("-opt:noref");
341ece8a530Spatrick
3421cf9926bSpatrick if (args.hasFlag(OPT_demangle, OPT_no_demangle, true))
3431cf9926bSpatrick add("-demangle");
3441cf9926bSpatrick else
3451cf9926bSpatrick add("-demangle:no");
3461cf9926bSpatrick
347bb684c34Spatrick if (args.hasFlag(OPT_enable_auto_import, OPT_disable_auto_import, true))
348bb684c34Spatrick add("-auto-import");
349bb684c34Spatrick else
350bb684c34Spatrick add("-auto-import:no");
351bb684c34Spatrick if (args.hasFlag(OPT_enable_runtime_pseudo_reloc,
352bb684c34Spatrick OPT_disable_runtime_pseudo_reloc, true))
353bb684c34Spatrick add("-runtime-pseudo-reloc");
354bb684c34Spatrick else
355bb684c34Spatrick add("-runtime-pseudo-reloc:no");
356bb684c34Spatrick
3571cf9926bSpatrick if (args.hasFlag(OPT_allow_multiple_definition,
3581cf9926bSpatrick OPT_no_allow_multiple_definition, false))
3591cf9926bSpatrick add("-force:multiple");
3601cf9926bSpatrick
361ece8a530Spatrick if (auto *a = args.getLastArg(OPT_icf)) {
362ece8a530Spatrick StringRef s = a->getValue();
363ece8a530Spatrick if (s == "all")
364ece8a530Spatrick add("-opt:icf");
365ece8a530Spatrick else if (s == "safe" || s == "none")
366ece8a530Spatrick add("-opt:noicf");
367ece8a530Spatrick else
368ece8a530Spatrick error("unknown parameter: --icf=" + s);
369ece8a530Spatrick } else {
370ece8a530Spatrick add("-opt:noicf");
371ece8a530Spatrick }
372ece8a530Spatrick
373ece8a530Spatrick if (auto *a = args.getLastArg(OPT_m)) {
374ece8a530Spatrick StringRef s = a->getValue();
375ece8a530Spatrick if (s == "i386pe")
376ece8a530Spatrick add("-machine:x86");
377ece8a530Spatrick else if (s == "i386pep")
378ece8a530Spatrick add("-machine:x64");
379ece8a530Spatrick else if (s == "thumb2pe")
380ece8a530Spatrick add("-machine:arm");
381ece8a530Spatrick else if (s == "arm64pe")
382ece8a530Spatrick add("-machine:arm64");
383ece8a530Spatrick else
384ece8a530Spatrick error("unknown parameter: -m" + s);
385ece8a530Spatrick }
386ece8a530Spatrick
387*dfe94b16Srobert if (args.hasFlag(OPT_guard_cf, OPT_no_guard_cf, false)) {
388*dfe94b16Srobert if (args.hasFlag(OPT_guard_longjmp, OPT_no_guard_longjmp, true))
389*dfe94b16Srobert add("-guard:cf,longjmp");
390*dfe94b16Srobert else
391*dfe94b16Srobert add("-guard:cf,nolongjmp");
392*dfe94b16Srobert } else if (args.hasFlag(OPT_guard_longjmp, OPT_no_guard_longjmp, false)) {
393*dfe94b16Srobert auto *a = args.getLastArg(OPT_guard_longjmp);
394*dfe94b16Srobert warn("parameter " + a->getSpelling() +
395*dfe94b16Srobert " only takes effect when used with --guard-cf");
396*dfe94b16Srobert }
397*dfe94b16Srobert
398*dfe94b16Srobert if (auto *a = args.getLastArg(OPT_error_limit)) {
399*dfe94b16Srobert int n;
400*dfe94b16Srobert StringRef s = a->getValue();
401*dfe94b16Srobert if (s.getAsInteger(10, n))
402*dfe94b16Srobert error(a->getSpelling() + ": number expected, but got " + s);
403*dfe94b16Srobert else
404*dfe94b16Srobert add("-errorlimit:" + s);
405*dfe94b16Srobert }
406*dfe94b16Srobert
407ece8a530Spatrick for (auto *a : args.filtered(OPT_mllvm))
408ece8a530Spatrick add("-mllvm:" + StringRef(a->getValue()));
409ece8a530Spatrick
410ece8a530Spatrick for (auto *a : args.filtered(OPT_Xlink))
411ece8a530Spatrick add(a->getValue());
412ece8a530Spatrick
413ece8a530Spatrick if (args.getLastArgValue(OPT_m) == "i386pe")
414ece8a530Spatrick add("-alternatename:__image_base__=___ImageBase");
415ece8a530Spatrick else
416ece8a530Spatrick add("-alternatename:__image_base__=__ImageBase");
417ece8a530Spatrick
418ece8a530Spatrick for (auto *a : args.filtered(OPT_require_defined))
419ece8a530Spatrick add("-include:" + StringRef(a->getValue()));
420ece8a530Spatrick for (auto *a : args.filtered(OPT_undefined))
421ece8a530Spatrick add("-includeoptional:" + StringRef(a->getValue()));
422ece8a530Spatrick for (auto *a : args.filtered(OPT_delayload))
423ece8a530Spatrick add("-delayload:" + StringRef(a->getValue()));
4241cf9926bSpatrick for (auto *a : args.filtered(OPT_wrap))
4251cf9926bSpatrick add("-wrap:" + StringRef(a->getValue()));
426*dfe94b16Srobert for (auto *a : args.filtered(OPT_exclude_symbols))
427*dfe94b16Srobert add("-exclude-symbols:" + StringRef(a->getValue()));
428ece8a530Spatrick
429ece8a530Spatrick std::vector<StringRef> searchPaths;
430ece8a530Spatrick for (auto *a : args.filtered(OPT_L)) {
431ece8a530Spatrick searchPaths.push_back(a->getValue());
432ece8a530Spatrick add("-libpath:" + StringRef(a->getValue()));
433ece8a530Spatrick }
434ece8a530Spatrick
435ece8a530Spatrick StringRef prefix = "";
436ece8a530Spatrick bool isStatic = false;
437ece8a530Spatrick for (auto *a : args) {
438ece8a530Spatrick switch (a->getOption().getID()) {
439ece8a530Spatrick case OPT_INPUT:
4401cf9926bSpatrick if (StringRef(a->getValue()).endswith_insensitive(".def"))
441ece8a530Spatrick add("-def:" + StringRef(a->getValue()));
442ece8a530Spatrick else
443ece8a530Spatrick add(prefix + StringRef(a->getValue()));
444ece8a530Spatrick break;
445ece8a530Spatrick case OPT_l:
446ece8a530Spatrick add(prefix + searchLibrary(a->getValue(), searchPaths, isStatic));
447ece8a530Spatrick break;
448ece8a530Spatrick case OPT_whole_archive:
449ece8a530Spatrick prefix = "-wholearchive:";
450ece8a530Spatrick break;
451ece8a530Spatrick case OPT_no_whole_archive:
452ece8a530Spatrick prefix = "";
453ece8a530Spatrick break;
454ece8a530Spatrick case OPT_Bstatic:
455ece8a530Spatrick isStatic = true;
456ece8a530Spatrick break;
457ece8a530Spatrick case OPT_Bdynamic:
458ece8a530Spatrick isStatic = false;
459ece8a530Spatrick break;
460ece8a530Spatrick }
461ece8a530Spatrick }
462ece8a530Spatrick
463ece8a530Spatrick if (errorCount())
464ece8a530Spatrick return false;
465ece8a530Spatrick
466ece8a530Spatrick if (args.hasArg(OPT_verbose) || args.hasArg(OPT__HASH_HASH_HASH))
4671cf9926bSpatrick lld::errs() << llvm::join(linkArgs, " ") << "\n";
468ece8a530Spatrick
469ece8a530Spatrick if (args.hasArg(OPT__HASH_HASH_HASH))
470ece8a530Spatrick return true;
471ece8a530Spatrick
472ece8a530Spatrick // Repack vector of strings to vector of const char pointers for coff::link.
473ece8a530Spatrick std::vector<const char *> vec;
474ece8a530Spatrick for (const std::string &s : linkArgs)
475ece8a530Spatrick vec.push_back(s.c_str());
4761cf9926bSpatrick // Pass the actual binary name, to make error messages be printed with
4771cf9926bSpatrick // the right prefix.
4781cf9926bSpatrick vec[0] = argsArr[0];
479*dfe94b16Srobert
480*dfe94b16Srobert // The context will be re-created in the COFF driver.
481*dfe94b16Srobert lld::CommonLinkerContext::destroy();
482*dfe94b16Srobert
483*dfe94b16Srobert return coff::link(vec, stdoutOS, stderrOS, exitEarly, disableOutput);
484ece8a530Spatrick }
485