xref: /openbsd-src/gnu/llvm/lld/MinGW/Driver.cpp (revision dfe94b169149f14cc1aee2cf6dad58a8d9a1860c)
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