xref: /openbsd-src/gnu/llvm/clang/lib/Driver/ToolChains/Darwin.cpp (revision 12c855180aad702bbcca06e0398d774beeafb155)
1e5dd7070Spatrick //===--- Darwin.cpp - Darwin Tool and ToolChain Implementations -*- C++ -*-===//
2e5dd7070Spatrick //
3e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e5dd7070Spatrick //
7e5dd7070Spatrick //===----------------------------------------------------------------------===//
8e5dd7070Spatrick 
9e5dd7070Spatrick #include "Darwin.h"
10a9ac8606Spatrick #include "Arch/AArch64.h"
11e5dd7070Spatrick #include "Arch/ARM.h"
12e5dd7070Spatrick #include "CommonArgs.h"
13e5dd7070Spatrick #include "clang/Basic/AlignedAllocation.h"
14e5dd7070Spatrick #include "clang/Basic/ObjCRuntime.h"
15e5dd7070Spatrick #include "clang/Config/config.h"
16e5dd7070Spatrick #include "clang/Driver/Compilation.h"
17e5dd7070Spatrick #include "clang/Driver/Driver.h"
18e5dd7070Spatrick #include "clang/Driver/DriverDiagnostic.h"
19e5dd7070Spatrick #include "clang/Driver/Options.h"
20e5dd7070Spatrick #include "clang/Driver/SanitizerArgs.h"
21e5dd7070Spatrick #include "llvm/ADT/StringSwitch.h"
22e5dd7070Spatrick #include "llvm/Option/ArgList.h"
23e5dd7070Spatrick #include "llvm/ProfileData/InstrProf.h"
24e5dd7070Spatrick #include "llvm/Support/Path.h"
25e5dd7070Spatrick #include "llvm/Support/ScopedPrinter.h"
26e5dd7070Spatrick #include "llvm/Support/TargetParser.h"
27ec727ea7Spatrick #include "llvm/Support/Threading.h"
28e5dd7070Spatrick #include "llvm/Support/VirtualFileSystem.h"
29e5dd7070Spatrick #include <cstdlib> // ::getenv
30e5dd7070Spatrick 
31e5dd7070Spatrick using namespace clang::driver;
32e5dd7070Spatrick using namespace clang::driver::tools;
33e5dd7070Spatrick using namespace clang::driver::toolchains;
34e5dd7070Spatrick using namespace clang;
35e5dd7070Spatrick using namespace llvm::opt;
36e5dd7070Spatrick 
minimumMacCatalystDeploymentTarget()37*12c85518Srobert static VersionTuple minimumMacCatalystDeploymentTarget() {
38a9ac8606Spatrick   return VersionTuple(13, 1);
39a9ac8606Spatrick }
40a9ac8606Spatrick 
getArchTypeForMachOArchName(StringRef Str)41e5dd7070Spatrick llvm::Triple::ArchType darwin::getArchTypeForMachOArchName(StringRef Str) {
42e5dd7070Spatrick   // See arch(3) and llvm-gcc's driver-driver.c. We don't implement support for
43e5dd7070Spatrick   // archs which Darwin doesn't use.
44e5dd7070Spatrick 
45e5dd7070Spatrick   // The matching this routine does is fairly pointless, since it is neither the
46e5dd7070Spatrick   // complete architecture list, nor a reasonable subset. The problem is that
47*12c85518Srobert   // historically the driver accepts this and also ties its -march=
48e5dd7070Spatrick   // handling to the architecture name, so we need to be careful before removing
49e5dd7070Spatrick   // support for it.
50e5dd7070Spatrick 
51e5dd7070Spatrick   // This code must be kept in sync with Clang's Darwin specific argument
52e5dd7070Spatrick   // translation.
53e5dd7070Spatrick 
54e5dd7070Spatrick   return llvm::StringSwitch<llvm::Triple::ArchType>(Str)
55e5dd7070Spatrick       .Cases("ppc", "ppc601", "ppc603", "ppc604", "ppc604e", llvm::Triple::ppc)
56e5dd7070Spatrick       .Cases("ppc750", "ppc7400", "ppc7450", "ppc970", llvm::Triple::ppc)
57e5dd7070Spatrick       .Case("ppc64", llvm::Triple::ppc64)
58e5dd7070Spatrick       .Cases("i386", "i486", "i486SX", "i586", "i686", llvm::Triple::x86)
59e5dd7070Spatrick       .Cases("pentium", "pentpro", "pentIIm3", "pentIIm5", "pentium4",
60e5dd7070Spatrick              llvm::Triple::x86)
61e5dd7070Spatrick       .Cases("x86_64", "x86_64h", llvm::Triple::x86_64)
62*12c85518Srobert       // This is derived from the driver.
63e5dd7070Spatrick       .Cases("arm", "armv4t", "armv5", "armv6", "armv6m", llvm::Triple::arm)
64e5dd7070Spatrick       .Cases("armv7", "armv7em", "armv7k", "armv7m", llvm::Triple::arm)
65e5dd7070Spatrick       .Cases("armv7s", "xscale", llvm::Triple::arm)
66a9ac8606Spatrick       .Cases("arm64", "arm64e", llvm::Triple::aarch64)
67e5dd7070Spatrick       .Case("arm64_32", llvm::Triple::aarch64_32)
68e5dd7070Spatrick       .Case("r600", llvm::Triple::r600)
69e5dd7070Spatrick       .Case("amdgcn", llvm::Triple::amdgcn)
70e5dd7070Spatrick       .Case("nvptx", llvm::Triple::nvptx)
71e5dd7070Spatrick       .Case("nvptx64", llvm::Triple::nvptx64)
72e5dd7070Spatrick       .Case("amdil", llvm::Triple::amdil)
73e5dd7070Spatrick       .Case("spir", llvm::Triple::spir)
74e5dd7070Spatrick       .Default(llvm::Triple::UnknownArch);
75e5dd7070Spatrick }
76e5dd7070Spatrick 
setTripleTypeForMachOArchName(llvm::Triple & T,StringRef Str)77e5dd7070Spatrick void darwin::setTripleTypeForMachOArchName(llvm::Triple &T, StringRef Str) {
78e5dd7070Spatrick   const llvm::Triple::ArchType Arch = getArchTypeForMachOArchName(Str);
79e5dd7070Spatrick   llvm::ARM::ArchKind ArchKind = llvm::ARM::parseArch(Str);
80e5dd7070Spatrick   T.setArch(Arch);
81a9ac8606Spatrick   if (Arch != llvm::Triple::UnknownArch)
82e5dd7070Spatrick     T.setArchName(Str);
83a9ac8606Spatrick 
84a9ac8606Spatrick   if (ArchKind == llvm::ARM::ArchKind::ARMV6M ||
85e5dd7070Spatrick       ArchKind == llvm::ARM::ArchKind::ARMV7M ||
86e5dd7070Spatrick       ArchKind == llvm::ARM::ArchKind::ARMV7EM) {
87e5dd7070Spatrick     T.setOS(llvm::Triple::UnknownOS);
88e5dd7070Spatrick     T.setObjectFormat(llvm::Triple::MachO);
89e5dd7070Spatrick   }
90e5dd7070Spatrick }
91e5dd7070Spatrick 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const92e5dd7070Spatrick void darwin::Assembler::ConstructJob(Compilation &C, const JobAction &JA,
93e5dd7070Spatrick                                      const InputInfo &Output,
94e5dd7070Spatrick                                      const InputInfoList &Inputs,
95e5dd7070Spatrick                                      const ArgList &Args,
96e5dd7070Spatrick                                      const char *LinkingOutput) const {
97*12c85518Srobert   const llvm::Triple &T(getToolChain().getTriple());
98*12c85518Srobert 
99e5dd7070Spatrick   ArgStringList CmdArgs;
100e5dd7070Spatrick 
101e5dd7070Spatrick   assert(Inputs.size() == 1 && "Unexpected number of inputs.");
102e5dd7070Spatrick   const InputInfo &Input = Inputs[0];
103e5dd7070Spatrick 
104e5dd7070Spatrick   // Determine the original source input.
105e5dd7070Spatrick   const Action *SourceAction = &JA;
106e5dd7070Spatrick   while (SourceAction->getKind() != Action::InputClass) {
107e5dd7070Spatrick     assert(!SourceAction->getInputs().empty() && "unexpected root action!");
108e5dd7070Spatrick     SourceAction = SourceAction->getInputs()[0];
109e5dd7070Spatrick   }
110e5dd7070Spatrick 
111e5dd7070Spatrick   // If -fno-integrated-as is used add -Q to the darwin assembler driver to make
112e5dd7070Spatrick   // sure it runs its system assembler not clang's integrated assembler.
113e5dd7070Spatrick   // Applicable to darwin11+ and Xcode 4+.  darwin<10 lacked integrated-as.
114e5dd7070Spatrick   // FIXME: at run-time detect assembler capabilities or rely on version
115e5dd7070Spatrick   // information forwarded by -target-assembler-version.
116e5dd7070Spatrick   if (Args.hasArg(options::OPT_fno_integrated_as)) {
117e5dd7070Spatrick     if (!(T.isMacOSX() && T.isMacOSXVersionLT(10, 7)))
118e5dd7070Spatrick       CmdArgs.push_back("-Q");
119e5dd7070Spatrick   }
120e5dd7070Spatrick 
121e5dd7070Spatrick   // Forward -g, assuming we are dealing with an actual assembly file.
122e5dd7070Spatrick   if (SourceAction->getType() == types::TY_Asm ||
123e5dd7070Spatrick       SourceAction->getType() == types::TY_PP_Asm) {
124e5dd7070Spatrick     if (Args.hasArg(options::OPT_gstabs))
125e5dd7070Spatrick       CmdArgs.push_back("--gstabs");
126e5dd7070Spatrick     else if (Args.hasArg(options::OPT_g_Group))
127e5dd7070Spatrick       CmdArgs.push_back("-g");
128e5dd7070Spatrick   }
129e5dd7070Spatrick 
130e5dd7070Spatrick   // Derived from asm spec.
131e5dd7070Spatrick   AddMachOArch(Args, CmdArgs);
132e5dd7070Spatrick 
133e5dd7070Spatrick   // Use -force_cpusubtype_ALL on x86 by default.
134*12c85518Srobert   if (T.isX86() || Args.hasArg(options::OPT_force__cpusubtype__ALL))
135e5dd7070Spatrick     CmdArgs.push_back("-force_cpusubtype_ALL");
136e5dd7070Spatrick 
137e5dd7070Spatrick   if (getToolChain().getArch() != llvm::Triple::x86_64 &&
138e5dd7070Spatrick       (((Args.hasArg(options::OPT_mkernel) ||
139e5dd7070Spatrick          Args.hasArg(options::OPT_fapple_kext)) &&
140e5dd7070Spatrick         getMachOToolChain().isKernelStatic()) ||
141e5dd7070Spatrick        Args.hasArg(options::OPT_static)))
142e5dd7070Spatrick     CmdArgs.push_back("-static");
143e5dd7070Spatrick 
144e5dd7070Spatrick   Args.AddAllArgValues(CmdArgs, options::OPT_Wa_COMMA, options::OPT_Xassembler);
145e5dd7070Spatrick 
146e5dd7070Spatrick   assert(Output.isFilename() && "Unexpected lipo output.");
147e5dd7070Spatrick   CmdArgs.push_back("-o");
148e5dd7070Spatrick   CmdArgs.push_back(Output.getFilename());
149e5dd7070Spatrick 
150e5dd7070Spatrick   assert(Input.isFilename() && "Invalid input.");
151e5dd7070Spatrick   CmdArgs.push_back(Input.getFilename());
152e5dd7070Spatrick 
153e5dd7070Spatrick   // asm_final spec is empty.
154e5dd7070Spatrick 
155e5dd7070Spatrick   const char *Exec = Args.MakeArgString(getToolChain().GetProgramPath("as"));
156ec727ea7Spatrick   C.addCommand(std::make_unique<Command>(JA, *this, ResponseFileSupport::None(),
157a9ac8606Spatrick                                          Exec, CmdArgs, Inputs, Output));
158e5dd7070Spatrick }
159e5dd7070Spatrick 
anchor()160e5dd7070Spatrick void darwin::MachOTool::anchor() {}
161e5dd7070Spatrick 
AddMachOArch(const ArgList & Args,ArgStringList & CmdArgs) const162e5dd7070Spatrick void darwin::MachOTool::AddMachOArch(const ArgList &Args,
163e5dd7070Spatrick                                      ArgStringList &CmdArgs) const {
164e5dd7070Spatrick   StringRef ArchName = getMachOToolChain().getMachOArchName(Args);
165e5dd7070Spatrick 
166e5dd7070Spatrick   // Derived from darwin_arch spec.
167e5dd7070Spatrick   CmdArgs.push_back("-arch");
168e5dd7070Spatrick   CmdArgs.push_back(Args.MakeArgString(ArchName));
169e5dd7070Spatrick 
170e5dd7070Spatrick   // FIXME: Is this needed anymore?
171e5dd7070Spatrick   if (ArchName == "arm")
172e5dd7070Spatrick     CmdArgs.push_back("-force_cpusubtype_ALL");
173e5dd7070Spatrick }
174e5dd7070Spatrick 
NeedsTempPath(const InputInfoList & Inputs) const175e5dd7070Spatrick bool darwin::Linker::NeedsTempPath(const InputInfoList &Inputs) const {
176e5dd7070Spatrick   // We only need to generate a temp path for LTO if we aren't compiling object
177e5dd7070Spatrick   // files. When compiling source files, we run 'dsymutil' after linking. We
178e5dd7070Spatrick   // don't run 'dsymutil' when compiling object files.
179e5dd7070Spatrick   for (const auto &Input : Inputs)
180e5dd7070Spatrick     if (Input.getType() != types::TY_Object)
181e5dd7070Spatrick       return true;
182e5dd7070Spatrick 
183e5dd7070Spatrick   return false;
184e5dd7070Spatrick }
185e5dd7070Spatrick 
186e5dd7070Spatrick /// Pass -no_deduplicate to ld64 under certain conditions:
187e5dd7070Spatrick ///
188e5dd7070Spatrick /// - Either -O0 or -O1 is explicitly specified
189e5dd7070Spatrick /// - No -O option is specified *and* this is a compile+link (implicit -O0)
190e5dd7070Spatrick ///
191e5dd7070Spatrick /// Also do *not* add -no_deduplicate when no -O option is specified and this
192e5dd7070Spatrick /// is just a link (we can't imply -O0)
shouldLinkerNotDedup(bool IsLinkerOnlyAction,const ArgList & Args)193e5dd7070Spatrick static bool shouldLinkerNotDedup(bool IsLinkerOnlyAction, const ArgList &Args) {
194e5dd7070Spatrick   if (Arg *A = Args.getLastArg(options::OPT_O_Group)) {
195e5dd7070Spatrick     if (A->getOption().matches(options::OPT_O0))
196e5dd7070Spatrick       return true;
197e5dd7070Spatrick     if (A->getOption().matches(options::OPT_O))
198e5dd7070Spatrick       return llvm::StringSwitch<bool>(A->getValue())
199e5dd7070Spatrick                     .Case("1", true)
200e5dd7070Spatrick                     .Default(false);
201e5dd7070Spatrick     return false; // OPT_Ofast & OPT_O4
202e5dd7070Spatrick   }
203e5dd7070Spatrick 
204e5dd7070Spatrick   if (!IsLinkerOnlyAction) // Implicit -O0 for compile+linker only.
205e5dd7070Spatrick     return true;
206e5dd7070Spatrick   return false;
207e5dd7070Spatrick }
208e5dd7070Spatrick 
AddLinkArgs(Compilation & C,const ArgList & Args,ArgStringList & CmdArgs,const InputInfoList & Inputs,VersionTuple Version,bool LinkerIsLLD) const209e5dd7070Spatrick void darwin::Linker::AddLinkArgs(Compilation &C, const ArgList &Args,
210e5dd7070Spatrick                                  ArgStringList &CmdArgs,
211ec727ea7Spatrick                                  const InputInfoList &Inputs,
212*12c85518Srobert                                  VersionTuple Version, bool LinkerIsLLD) const {
213e5dd7070Spatrick   const Driver &D = getToolChain().getDriver();
214e5dd7070Spatrick   const toolchains::MachO &MachOTC = getMachOToolChain();
215e5dd7070Spatrick 
216e5dd7070Spatrick   // Newer linkers support -demangle. Pass it if supported and not disabled by
217e5dd7070Spatrick   // the user.
218*12c85518Srobert   if ((Version >= VersionTuple(100) || LinkerIsLLD) &&
219a9ac8606Spatrick       !Args.hasArg(options::OPT_Z_Xlinker__no_demangle))
220e5dd7070Spatrick     CmdArgs.push_back("-demangle");
221e5dd7070Spatrick 
222*12c85518Srobert   if (Args.hasArg(options::OPT_rdynamic) &&
223*12c85518Srobert       (Version >= VersionTuple(137) || LinkerIsLLD))
224e5dd7070Spatrick     CmdArgs.push_back("-export_dynamic");
225e5dd7070Spatrick 
226e5dd7070Spatrick   // If we are using App Extension restrictions, pass a flag to the linker
227e5dd7070Spatrick   // telling it that the compiled code has been audited.
228e5dd7070Spatrick   if (Args.hasFlag(options::OPT_fapplication_extension,
229e5dd7070Spatrick                    options::OPT_fno_application_extension, false))
230e5dd7070Spatrick     CmdArgs.push_back("-application_extension");
231e5dd7070Spatrick 
232*12c85518Srobert   if (D.isUsingLTO() && (Version >= VersionTuple(116) || LinkerIsLLD) &&
233*12c85518Srobert       NeedsTempPath(Inputs)) {
234e5dd7070Spatrick     std::string TmpPathName;
235e5dd7070Spatrick     if (D.getLTOMode() == LTOK_Full) {
236e5dd7070Spatrick       // If we are using full LTO, then automatically create a temporary file
237e5dd7070Spatrick       // path for the linker to use, so that it's lifetime will extend past a
238e5dd7070Spatrick       // possible dsymutil step.
239e5dd7070Spatrick       TmpPathName =
240e5dd7070Spatrick           D.GetTemporaryPath("cc", types::getTypeTempSuffix(types::TY_Object));
241e5dd7070Spatrick     } else if (D.getLTOMode() == LTOK_Thin)
242e5dd7070Spatrick       // If we are using thin LTO, then create a directory instead.
243e5dd7070Spatrick       TmpPathName = D.GetTemporaryDirectory("thinlto");
244e5dd7070Spatrick 
245e5dd7070Spatrick     if (!TmpPathName.empty()) {
246e5dd7070Spatrick       auto *TmpPath = C.getArgs().MakeArgString(TmpPathName);
247e5dd7070Spatrick       C.addTempFile(TmpPath);
248e5dd7070Spatrick       CmdArgs.push_back("-object_path_lto");
249e5dd7070Spatrick       CmdArgs.push_back(TmpPath);
250e5dd7070Spatrick     }
251e5dd7070Spatrick   }
252e5dd7070Spatrick 
253e5dd7070Spatrick   // Use -lto_library option to specify the libLTO.dylib path. Try to find
254e5dd7070Spatrick   // it in clang installed libraries. ld64 will only look at this argument
255e5dd7070Spatrick   // when it actually uses LTO, so libLTO.dylib only needs to exist at link
256e5dd7070Spatrick   // time if ld64 decides that it needs to use LTO.
257e5dd7070Spatrick   // Since this is passed unconditionally, ld64 will never look for libLTO.dylib
258e5dd7070Spatrick   // next to it. That's ok since ld64 using a libLTO.dylib not matching the
259e5dd7070Spatrick   // clang version won't work anyways.
260a9ac8606Spatrick   // lld is built at the same revision as clang and statically links in
261a9ac8606Spatrick   // LLVM libraries, so it doesn't need libLTO.dylib.
262*12c85518Srobert   if (Version >= VersionTuple(133) && !LinkerIsLLD) {
263e5dd7070Spatrick     // Search for libLTO in <InstalledDir>/../lib/libLTO.dylib
264e5dd7070Spatrick     StringRef P = llvm::sys::path::parent_path(D.Dir);
265e5dd7070Spatrick     SmallString<128> LibLTOPath(P);
266e5dd7070Spatrick     llvm::sys::path::append(LibLTOPath, "lib");
267e5dd7070Spatrick     llvm::sys::path::append(LibLTOPath, "libLTO.dylib");
268e5dd7070Spatrick     CmdArgs.push_back("-lto_library");
269e5dd7070Spatrick     CmdArgs.push_back(C.getArgs().MakeArgString(LibLTOPath));
270e5dd7070Spatrick   }
271e5dd7070Spatrick 
272*12c85518Srobert   // ld64 version 262 and above runs the deduplicate pass by default.
273*12c85518Srobert   // FIXME: lld doesn't dedup by default. Should we pass `--icf=safe`
274*12c85518Srobert   //        if `!shouldLinkerNotDedup()` if LinkerIsLLD here?
275*12c85518Srobert   if (Version >= VersionTuple(262) &&
276*12c85518Srobert       shouldLinkerNotDedup(C.getJobs().empty(), Args))
277e5dd7070Spatrick     CmdArgs.push_back("-no_deduplicate");
278e5dd7070Spatrick 
279e5dd7070Spatrick   // Derived from the "link" spec.
280e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_static);
281e5dd7070Spatrick   if (!Args.hasArg(options::OPT_static))
282e5dd7070Spatrick     CmdArgs.push_back("-dynamic");
283e5dd7070Spatrick   if (Args.hasArg(options::OPT_fgnu_runtime)) {
284e5dd7070Spatrick     // FIXME: gcc replaces -lobjc in forward args with -lobjc-gnu
285e5dd7070Spatrick     // here. How do we wish to handle such things?
286e5dd7070Spatrick   }
287e5dd7070Spatrick 
288e5dd7070Spatrick   if (!Args.hasArg(options::OPT_dynamiclib)) {
289e5dd7070Spatrick     AddMachOArch(Args, CmdArgs);
290e5dd7070Spatrick     // FIXME: Why do this only on this path?
291e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_force__cpusubtype__ALL);
292e5dd7070Spatrick 
293e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_bundle);
294e5dd7070Spatrick     Args.AddAllArgs(CmdArgs, options::OPT_bundle__loader);
295e5dd7070Spatrick     Args.AddAllArgs(CmdArgs, options::OPT_client__name);
296e5dd7070Spatrick 
297e5dd7070Spatrick     Arg *A;
298e5dd7070Spatrick     if ((A = Args.getLastArg(options::OPT_compatibility__version)) ||
299e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_current__version)) ||
300e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_install__name)))
301e5dd7070Spatrick       D.Diag(diag::err_drv_argument_only_allowed_with) << A->getAsString(Args)
302e5dd7070Spatrick                                                        << "-dynamiclib";
303e5dd7070Spatrick 
304e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_force__flat__namespace);
305e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_keep__private__externs);
306e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_private__bundle);
307e5dd7070Spatrick   } else {
308e5dd7070Spatrick     CmdArgs.push_back("-dylib");
309e5dd7070Spatrick 
310e5dd7070Spatrick     Arg *A;
311e5dd7070Spatrick     if ((A = Args.getLastArg(options::OPT_bundle)) ||
312e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_bundle__loader)) ||
313e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_client__name)) ||
314e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_force__flat__namespace)) ||
315e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_keep__private__externs)) ||
316e5dd7070Spatrick         (A = Args.getLastArg(options::OPT_private__bundle)))
317e5dd7070Spatrick       D.Diag(diag::err_drv_argument_not_allowed_with) << A->getAsString(Args)
318e5dd7070Spatrick                                                       << "-dynamiclib";
319e5dd7070Spatrick 
320e5dd7070Spatrick     Args.AddAllArgsTranslated(CmdArgs, options::OPT_compatibility__version,
321e5dd7070Spatrick                               "-dylib_compatibility_version");
322e5dd7070Spatrick     Args.AddAllArgsTranslated(CmdArgs, options::OPT_current__version,
323e5dd7070Spatrick                               "-dylib_current_version");
324e5dd7070Spatrick 
325e5dd7070Spatrick     AddMachOArch(Args, CmdArgs);
326e5dd7070Spatrick 
327e5dd7070Spatrick     Args.AddAllArgsTranslated(CmdArgs, options::OPT_install__name,
328e5dd7070Spatrick                               "-dylib_install_name");
329e5dd7070Spatrick   }
330e5dd7070Spatrick 
331e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_all__load);
332e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_allowable__client);
333e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_bind__at__load);
334e5dd7070Spatrick   if (MachOTC.isTargetIOSBased())
335e5dd7070Spatrick     Args.AddLastArg(CmdArgs, options::OPT_arch__errors__fatal);
336e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_dead__strip);
337e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_no__dead__strip__inits__and__terms);
338e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_dylib__file);
339e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_dynamic);
340e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_exported__symbols__list);
341e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_flat__namespace);
342e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_force__load);
343e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_headerpad__max__install__names);
344e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_image__base);
345e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_init);
346e5dd7070Spatrick 
347e5dd7070Spatrick   // Add the deployment target.
348*12c85518Srobert   if (Version >= VersionTuple(520) || LinkerIsLLD)
349e5dd7070Spatrick     MachOTC.addPlatformVersionArgs(Args, CmdArgs);
350e5dd7070Spatrick   else
351e5dd7070Spatrick     MachOTC.addMinVersionArgs(Args, CmdArgs);
352e5dd7070Spatrick 
353e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_nomultidefs);
354e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_multi__module);
355e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_single__module);
356e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_multiply__defined);
357e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_multiply__defined__unused);
358e5dd7070Spatrick 
359e5dd7070Spatrick   if (const Arg *A =
360e5dd7070Spatrick           Args.getLastArg(options::OPT_fpie, options::OPT_fPIE,
361e5dd7070Spatrick                           options::OPT_fno_pie, options::OPT_fno_PIE)) {
362e5dd7070Spatrick     if (A->getOption().matches(options::OPT_fpie) ||
363e5dd7070Spatrick         A->getOption().matches(options::OPT_fPIE))
364e5dd7070Spatrick       CmdArgs.push_back("-pie");
365e5dd7070Spatrick     else
366e5dd7070Spatrick       CmdArgs.push_back("-no_pie");
367e5dd7070Spatrick   }
368e5dd7070Spatrick 
369e5dd7070Spatrick   // for embed-bitcode, use -bitcode_bundle in linker command
370e5dd7070Spatrick   if (C.getDriver().embedBitcodeEnabled()) {
371e5dd7070Spatrick     // Check if the toolchain supports bitcode build flow.
372e5dd7070Spatrick     if (MachOTC.SupportsEmbeddedBitcode()) {
373e5dd7070Spatrick       CmdArgs.push_back("-bitcode_bundle");
374*12c85518Srobert       // FIXME: Pass this if LinkerIsLLD too, once it implements this flag.
375*12c85518Srobert       if (C.getDriver().embedBitcodeMarkerOnly() &&
376*12c85518Srobert           Version >= VersionTuple(278)) {
377e5dd7070Spatrick         CmdArgs.push_back("-bitcode_process_mode");
378e5dd7070Spatrick         CmdArgs.push_back("marker");
379e5dd7070Spatrick       }
380e5dd7070Spatrick     } else
381e5dd7070Spatrick       D.Diag(diag::err_drv_bitcode_unsupported_on_toolchain);
382e5dd7070Spatrick   }
383e5dd7070Spatrick 
384a9ac8606Spatrick   // If GlobalISel is enabled, pass it through to LLVM.
385a9ac8606Spatrick   if (Arg *A = Args.getLastArg(options::OPT_fglobal_isel,
386a9ac8606Spatrick                                options::OPT_fno_global_isel)) {
387a9ac8606Spatrick     if (A->getOption().matches(options::OPT_fglobal_isel)) {
388a9ac8606Spatrick       CmdArgs.push_back("-mllvm");
389a9ac8606Spatrick       CmdArgs.push_back("-global-isel");
390a9ac8606Spatrick       // Disable abort and fall back to SDAG silently.
391a9ac8606Spatrick       CmdArgs.push_back("-mllvm");
392a9ac8606Spatrick       CmdArgs.push_back("-global-isel-abort=0");
393a9ac8606Spatrick     }
394a9ac8606Spatrick   }
395a9ac8606Spatrick 
396e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_prebind);
397e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_noprebind);
398e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_nofixprebinding);
399e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_prebind__all__twolevel__modules);
400e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_read__only__relocs);
401e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sectcreate);
402e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sectorder);
403e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_seg1addr);
404e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segprot);
405e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segaddr);
406e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segs__read__only__addr);
407e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segs__read__write__addr);
408e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_seg__addr__table);
409e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_seg__addr__table__filename);
410e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sub__library);
411e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sub__umbrella);
412e5dd7070Spatrick 
413e5dd7070Spatrick   // Give --sysroot= preference, over the Apple specific behavior to also use
414e5dd7070Spatrick   // --isysroot as the syslibroot.
415e5dd7070Spatrick   StringRef sysroot = C.getSysRoot();
416e5dd7070Spatrick   if (sysroot != "") {
417e5dd7070Spatrick     CmdArgs.push_back("-syslibroot");
418e5dd7070Spatrick     CmdArgs.push_back(C.getArgs().MakeArgString(sysroot));
419e5dd7070Spatrick   } else if (const Arg *A = Args.getLastArg(options::OPT_isysroot)) {
420e5dd7070Spatrick     CmdArgs.push_back("-syslibroot");
421e5dd7070Spatrick     CmdArgs.push_back(A->getValue());
422e5dd7070Spatrick   }
423e5dd7070Spatrick 
424e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_twolevel__namespace);
425e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_twolevel__namespace__hints);
426e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_umbrella);
427e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_undefined);
428e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_unexported__symbols__list);
429e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_weak__reference__mismatches);
430e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_X_Flag);
431e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_y);
432e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_w);
433e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_pagezero__size);
434e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segs__read__);
435e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_seglinkedit);
436e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_noseglinkedit);
437e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sectalign);
438e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_sectobjectsymbols);
439e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_segcreate);
440a9ac8606Spatrick   Args.AddLastArg(CmdArgs, options::OPT_why_load);
441e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_whatsloaded);
442e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_dylinker__install__name);
443e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_dylinker);
444e5dd7070Spatrick   Args.AddLastArg(CmdArgs, options::OPT_Mach);
445e5dd7070Spatrick }
446e5dd7070Spatrick 
447e5dd7070Spatrick /// Determine whether we are linking the ObjC runtime.
isObjCRuntimeLinked(const ArgList & Args)448e5dd7070Spatrick static bool isObjCRuntimeLinked(const ArgList &Args) {
449e5dd7070Spatrick   if (isObjCAutoRefCount(Args)) {
450e5dd7070Spatrick     Args.ClaimAllArgs(options::OPT_fobjc_link_runtime);
451e5dd7070Spatrick     return true;
452e5dd7070Spatrick   }
453e5dd7070Spatrick   return Args.hasArg(options::OPT_fobjc_link_runtime);
454e5dd7070Spatrick }
455e5dd7070Spatrick 
checkRemarksOptions(const Driver & D,const ArgList & Args,const llvm::Triple & Triple)456ec727ea7Spatrick static bool checkRemarksOptions(const Driver &D, const ArgList &Args,
457ec727ea7Spatrick                                 const llvm::Triple &Triple) {
458ec727ea7Spatrick   // When enabling remarks, we need to error if:
459ec727ea7Spatrick   // * The remark file is specified but we're targeting multiple architectures,
460ec727ea7Spatrick   // which means more than one remark file is being generated.
461ec727ea7Spatrick   bool hasMultipleInvocations =
462ec727ea7Spatrick       Args.getAllArgValues(options::OPT_arch).size() > 1;
463ec727ea7Spatrick   bool hasExplicitOutputFile =
464ec727ea7Spatrick       Args.getLastArg(options::OPT_foptimization_record_file_EQ);
465ec727ea7Spatrick   if (hasMultipleInvocations && hasExplicitOutputFile) {
466ec727ea7Spatrick     D.Diag(diag::err_drv_invalid_output_with_multiple_archs)
467ec727ea7Spatrick         << "-foptimization-record-file";
468ec727ea7Spatrick     return false;
469ec727ea7Spatrick   }
470ec727ea7Spatrick   return true;
471ec727ea7Spatrick }
472ec727ea7Spatrick 
renderRemarksOptions(const ArgList & Args,ArgStringList & CmdArgs,const llvm::Triple & Triple,const InputInfo & Output,const JobAction & JA)473ec727ea7Spatrick static void renderRemarksOptions(const ArgList &Args, ArgStringList &CmdArgs,
474ec727ea7Spatrick                                  const llvm::Triple &Triple,
475ec727ea7Spatrick                                  const InputInfo &Output, const JobAction &JA) {
476ec727ea7Spatrick   StringRef Format = "yaml";
477ec727ea7Spatrick   if (const Arg *A = Args.getLastArg(options::OPT_fsave_optimization_record_EQ))
478ec727ea7Spatrick     Format = A->getValue();
479ec727ea7Spatrick 
480ec727ea7Spatrick   CmdArgs.push_back("-mllvm");
481ec727ea7Spatrick   CmdArgs.push_back("-lto-pass-remarks-output");
482ec727ea7Spatrick   CmdArgs.push_back("-mllvm");
483ec727ea7Spatrick 
484ec727ea7Spatrick   const Arg *A = Args.getLastArg(options::OPT_foptimization_record_file_EQ);
485ec727ea7Spatrick   if (A) {
486ec727ea7Spatrick     CmdArgs.push_back(A->getValue());
487ec727ea7Spatrick   } else {
488ec727ea7Spatrick     assert(Output.isFilename() && "Unexpected ld output.");
489ec727ea7Spatrick     SmallString<128> F;
490ec727ea7Spatrick     F = Output.getFilename();
491ec727ea7Spatrick     F += ".opt.";
492ec727ea7Spatrick     F += Format;
493ec727ea7Spatrick 
494ec727ea7Spatrick     CmdArgs.push_back(Args.MakeArgString(F));
495ec727ea7Spatrick   }
496ec727ea7Spatrick 
497ec727ea7Spatrick   if (const Arg *A =
498ec727ea7Spatrick           Args.getLastArg(options::OPT_foptimization_record_passes_EQ)) {
499ec727ea7Spatrick     CmdArgs.push_back("-mllvm");
500ec727ea7Spatrick     std::string Passes =
501ec727ea7Spatrick         std::string("-lto-pass-remarks-filter=") + A->getValue();
502ec727ea7Spatrick     CmdArgs.push_back(Args.MakeArgString(Passes));
503ec727ea7Spatrick   }
504ec727ea7Spatrick 
505ec727ea7Spatrick   if (!Format.empty()) {
506ec727ea7Spatrick     CmdArgs.push_back("-mllvm");
507ec727ea7Spatrick     Twine FormatArg = Twine("-lto-pass-remarks-format=") + Format;
508ec727ea7Spatrick     CmdArgs.push_back(Args.MakeArgString(FormatArg));
509ec727ea7Spatrick   }
510ec727ea7Spatrick 
511ec727ea7Spatrick   if (getLastProfileUseArg(Args)) {
512ec727ea7Spatrick     CmdArgs.push_back("-mllvm");
513ec727ea7Spatrick     CmdArgs.push_back("-lto-pass-remarks-with-hotness");
514ec727ea7Spatrick 
515ec727ea7Spatrick     if (const Arg *A =
516ec727ea7Spatrick             Args.getLastArg(options::OPT_fdiagnostics_hotness_threshold_EQ)) {
517ec727ea7Spatrick       CmdArgs.push_back("-mllvm");
518ec727ea7Spatrick       std::string Opt =
519ec727ea7Spatrick           std::string("-lto-pass-remarks-hotness-threshold=") + A->getValue();
520ec727ea7Spatrick       CmdArgs.push_back(Args.MakeArgString(Opt));
521ec727ea7Spatrick     }
522ec727ea7Spatrick   }
523ec727ea7Spatrick }
524ec727ea7Spatrick 
525*12c85518Srobert static void AppendPlatformPrefix(SmallString<128> &Path, const llvm::Triple &T);
526*12c85518Srobert 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const527e5dd7070Spatrick void darwin::Linker::ConstructJob(Compilation &C, const JobAction &JA,
528e5dd7070Spatrick                                   const InputInfo &Output,
529e5dd7070Spatrick                                   const InputInfoList &Inputs,
530e5dd7070Spatrick                                   const ArgList &Args,
531e5dd7070Spatrick                                   const char *LinkingOutput) const {
532e5dd7070Spatrick   assert(Output.getType() == types::TY_Image && "Invalid linker output type.");
533e5dd7070Spatrick 
534e5dd7070Spatrick   // If the number of arguments surpasses the system limits, we will encode the
535e5dd7070Spatrick   // input files in a separate file, shortening the command line. To this end,
536e5dd7070Spatrick   // build a list of input file names that can be passed via a file with the
537e5dd7070Spatrick   // -filelist linker option.
538e5dd7070Spatrick   llvm::opt::ArgStringList InputFileList;
539e5dd7070Spatrick 
540e5dd7070Spatrick   // The logic here is derived from gcc's behavior; most of which
541e5dd7070Spatrick   // comes from specs (starting with link_command). Consult gcc for
542e5dd7070Spatrick   // more information.
543e5dd7070Spatrick   ArgStringList CmdArgs;
544e5dd7070Spatrick 
545e5dd7070Spatrick   /// Hack(tm) to ignore linking errors when we are doing ARC migration.
546e5dd7070Spatrick   if (Args.hasArg(options::OPT_ccc_arcmt_check,
547e5dd7070Spatrick                   options::OPT_ccc_arcmt_migrate)) {
548e5dd7070Spatrick     for (const auto &Arg : Args)
549e5dd7070Spatrick       Arg->claim();
550e5dd7070Spatrick     const char *Exec =
551e5dd7070Spatrick         Args.MakeArgString(getToolChain().GetProgramPath("touch"));
552e5dd7070Spatrick     CmdArgs.push_back(Output.getFilename());
553*12c85518Srobert     C.addCommand(std::make_unique<Command>(JA, *this,
554*12c85518Srobert                                            ResponseFileSupport::None(), Exec,
555*12c85518Srobert                                            CmdArgs, std::nullopt, Output));
556e5dd7070Spatrick     return;
557e5dd7070Spatrick   }
558e5dd7070Spatrick 
559*12c85518Srobert   VersionTuple Version = getMachOToolChain().getLinkerVersion(Args);
560ec727ea7Spatrick 
561*12c85518Srobert   bool LinkerIsLLD;
562*12c85518Srobert   const char *Exec =
563*12c85518Srobert       Args.MakeArgString(getToolChain().GetLinkerPath(&LinkerIsLLD));
564a9ac8606Spatrick 
565e5dd7070Spatrick   // I'm not sure why this particular decomposition exists in gcc, but
566e5dd7070Spatrick   // we follow suite for ease of comparison.
567*12c85518Srobert   AddLinkArgs(C, Args, CmdArgs, Inputs, Version, LinkerIsLLD);
568e5dd7070Spatrick 
569ec727ea7Spatrick   if (willEmitRemarks(Args) &&
570ec727ea7Spatrick       checkRemarksOptions(getToolChain().getDriver(), Args,
571ec727ea7Spatrick                           getToolChain().getTriple()))
572ec727ea7Spatrick     renderRemarksOptions(Args, CmdArgs, getToolChain().getTriple(), Output, JA);
573e5dd7070Spatrick 
574e5dd7070Spatrick   // Propagate the -moutline flag to the linker in LTO.
575e5dd7070Spatrick   if (Arg *A =
576e5dd7070Spatrick           Args.getLastArg(options::OPT_moutline, options::OPT_mno_outline)) {
577e5dd7070Spatrick     if (A->getOption().matches(options::OPT_moutline)) {
578e5dd7070Spatrick       if (getMachOToolChain().getMachOArchName(Args) == "arm64") {
579e5dd7070Spatrick         CmdArgs.push_back("-mllvm");
580e5dd7070Spatrick         CmdArgs.push_back("-enable-machine-outliner");
581e5dd7070Spatrick 
582e5dd7070Spatrick         // Outline from linkonceodr functions by default in LTO.
583e5dd7070Spatrick         CmdArgs.push_back("-mllvm");
584e5dd7070Spatrick         CmdArgs.push_back("-enable-linkonceodr-outlining");
585e5dd7070Spatrick       }
586e5dd7070Spatrick     } else {
587e5dd7070Spatrick       // Disable all outlining behaviour if we have mno-outline. We need to do
588e5dd7070Spatrick       // this explicitly, because targets which support default outlining will
589e5dd7070Spatrick       // try to do work if we don't.
590e5dd7070Spatrick       CmdArgs.push_back("-mllvm");
591e5dd7070Spatrick       CmdArgs.push_back("-enable-machine-outliner=never");
592e5dd7070Spatrick     }
593e5dd7070Spatrick   }
594e5dd7070Spatrick 
595e5dd7070Spatrick   // Setup statistics file output.
596e5dd7070Spatrick   SmallString<128> StatsFile =
597e5dd7070Spatrick       getStatsFileName(Args, Output, Inputs[0], getToolChain().getDriver());
598e5dd7070Spatrick   if (!StatsFile.empty()) {
599e5dd7070Spatrick     CmdArgs.push_back("-mllvm");
600e5dd7070Spatrick     CmdArgs.push_back(Args.MakeArgString("-lto-stats-file=" + StatsFile.str()));
601e5dd7070Spatrick   }
602e5dd7070Spatrick 
603e5dd7070Spatrick   // It seems that the 'e' option is completely ignored for dynamic executables
604e5dd7070Spatrick   // (the default), and with static executables, the last one wins, as expected.
605e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, {options::OPT_d_Flag, options::OPT_s, options::OPT_t,
606e5dd7070Spatrick                             options::OPT_Z_Flag, options::OPT_u_Group,
607e5dd7070Spatrick                             options::OPT_e, options::OPT_r});
608e5dd7070Spatrick 
609e5dd7070Spatrick   // Forward -ObjC when either -ObjC or -ObjC++ is used, to force loading
610e5dd7070Spatrick   // members of static archive libraries which implement Objective-C classes or
611e5dd7070Spatrick   // categories.
612e5dd7070Spatrick   if (Args.hasArg(options::OPT_ObjC) || Args.hasArg(options::OPT_ObjCXX))
613e5dd7070Spatrick     CmdArgs.push_back("-ObjC");
614e5dd7070Spatrick 
615e5dd7070Spatrick   CmdArgs.push_back("-o");
616e5dd7070Spatrick   CmdArgs.push_back(Output.getFilename());
617e5dd7070Spatrick 
618e5dd7070Spatrick   if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles))
619e5dd7070Spatrick     getMachOToolChain().addStartObjectFileArgs(Args, CmdArgs);
620e5dd7070Spatrick 
621e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_L);
622e5dd7070Spatrick 
623e5dd7070Spatrick   AddLinkerInputs(getToolChain(), Inputs, Args, CmdArgs, JA);
624e5dd7070Spatrick   // Build the input file for -filelist (list of linker input files) in case we
625e5dd7070Spatrick   // need it later
626e5dd7070Spatrick   for (const auto &II : Inputs) {
627e5dd7070Spatrick     if (!II.isFilename()) {
628e5dd7070Spatrick       // This is a linker input argument.
629e5dd7070Spatrick       // We cannot mix input arguments and file names in a -filelist input, thus
630e5dd7070Spatrick       // we prematurely stop our list (remaining files shall be passed as
631e5dd7070Spatrick       // arguments).
632e5dd7070Spatrick       if (InputFileList.size() > 0)
633e5dd7070Spatrick         break;
634e5dd7070Spatrick 
635e5dd7070Spatrick       continue;
636e5dd7070Spatrick     }
637e5dd7070Spatrick 
638e5dd7070Spatrick     InputFileList.push_back(II.getFilename());
639e5dd7070Spatrick   }
640e5dd7070Spatrick 
641*12c85518Srobert   // Additional linker set-up and flags for Fortran. This is required in order
642*12c85518Srobert   // to generate executables.
643*12c85518Srobert   if (getToolChain().getDriver().IsFlangMode()) {
644*12c85518Srobert     addFortranRuntimeLibraryPath(getToolChain(), Args, CmdArgs);
645*12c85518Srobert     addFortranRuntimeLibs(getToolChain(), CmdArgs);
646*12c85518Srobert   }
647*12c85518Srobert 
648e5dd7070Spatrick   if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs))
649e5dd7070Spatrick     addOpenMPRuntime(CmdArgs, getToolChain(), Args);
650e5dd7070Spatrick 
651e5dd7070Spatrick   if (isObjCRuntimeLinked(Args) &&
652e5dd7070Spatrick       !Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
653e5dd7070Spatrick     // We use arclite library for both ARC and subscripting support.
654e5dd7070Spatrick     getMachOToolChain().AddLinkARCArgs(Args, CmdArgs);
655e5dd7070Spatrick 
656e5dd7070Spatrick     CmdArgs.push_back("-framework");
657e5dd7070Spatrick     CmdArgs.push_back("Foundation");
658e5dd7070Spatrick     // Link libobj.
659e5dd7070Spatrick     CmdArgs.push_back("-lobjc");
660e5dd7070Spatrick   }
661e5dd7070Spatrick 
662e5dd7070Spatrick   if (LinkingOutput) {
663e5dd7070Spatrick     CmdArgs.push_back("-arch_multiple");
664e5dd7070Spatrick     CmdArgs.push_back("-final_output");
665e5dd7070Spatrick     CmdArgs.push_back(LinkingOutput);
666e5dd7070Spatrick   }
667e5dd7070Spatrick 
668e5dd7070Spatrick   if (Args.hasArg(options::OPT_fnested_functions))
669e5dd7070Spatrick     CmdArgs.push_back("-allow_stack_execute");
670e5dd7070Spatrick 
671e5dd7070Spatrick   getMachOToolChain().addProfileRTLibs(Args, CmdArgs);
672e5dd7070Spatrick 
673ec727ea7Spatrick   StringRef Parallelism = getLTOParallelism(Args, getToolChain().getDriver());
674ec727ea7Spatrick   if (!Parallelism.empty()) {
675e5dd7070Spatrick     CmdArgs.push_back("-mllvm");
676ec727ea7Spatrick     unsigned NumThreads =
677ec727ea7Spatrick         llvm::get_threadpool_strategy(Parallelism)->compute_thread_count();
678ec727ea7Spatrick     CmdArgs.push_back(Args.MakeArgString("-threads=" + Twine(NumThreads)));
679e5dd7070Spatrick   }
680e5dd7070Spatrick 
681e5dd7070Spatrick   if (getToolChain().ShouldLinkCXXStdlib(Args))
682e5dd7070Spatrick     getToolChain().AddCXXStdlibLibArgs(Args, CmdArgs);
683e5dd7070Spatrick 
684e5dd7070Spatrick   bool NoStdOrDefaultLibs =
685e5dd7070Spatrick       Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs);
686e5dd7070Spatrick   bool ForceLinkBuiltins = Args.hasArg(options::OPT_fapple_link_rtlib);
687e5dd7070Spatrick   if (!NoStdOrDefaultLibs || ForceLinkBuiltins) {
688e5dd7070Spatrick     // link_ssp spec is empty.
689e5dd7070Spatrick 
690e5dd7070Spatrick     // If we have both -nostdlib/nodefaultlibs and -fapple-link-rtlib then
691e5dd7070Spatrick     // we just want to link the builtins, not the other libs like libSystem.
692e5dd7070Spatrick     if (NoStdOrDefaultLibs && ForceLinkBuiltins) {
693e5dd7070Spatrick       getMachOToolChain().AddLinkRuntimeLib(Args, CmdArgs, "builtins");
694e5dd7070Spatrick     } else {
695e5dd7070Spatrick       // Let the tool chain choose which runtime library to link.
696e5dd7070Spatrick       getMachOToolChain().AddLinkRuntimeLibArgs(Args, CmdArgs,
697e5dd7070Spatrick                                                 ForceLinkBuiltins);
698e5dd7070Spatrick 
699e5dd7070Spatrick       // No need to do anything for pthreads. Claim argument to avoid warning.
700e5dd7070Spatrick       Args.ClaimAllArgs(options::OPT_pthread);
701e5dd7070Spatrick       Args.ClaimAllArgs(options::OPT_pthreads);
702e5dd7070Spatrick     }
703e5dd7070Spatrick   }
704e5dd7070Spatrick 
705e5dd7070Spatrick   if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles)) {
706e5dd7070Spatrick     // endfile_spec is empty.
707e5dd7070Spatrick   }
708e5dd7070Spatrick 
709e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_T_Group);
710e5dd7070Spatrick   Args.AddAllArgs(CmdArgs, options::OPT_F);
711e5dd7070Spatrick 
712e5dd7070Spatrick   // -iframework should be forwarded as -F.
713e5dd7070Spatrick   for (const Arg *A : Args.filtered(options::OPT_iframework))
714e5dd7070Spatrick     CmdArgs.push_back(Args.MakeArgString(std::string("-F") + A->getValue()));
715e5dd7070Spatrick 
716e5dd7070Spatrick   if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) {
717e5dd7070Spatrick     if (Arg *A = Args.getLastArg(options::OPT_fveclib)) {
718e5dd7070Spatrick       if (A->getValue() == StringRef("Accelerate")) {
719e5dd7070Spatrick         CmdArgs.push_back("-framework");
720e5dd7070Spatrick         CmdArgs.push_back("Accelerate");
721e5dd7070Spatrick       }
722e5dd7070Spatrick     }
723e5dd7070Spatrick   }
724e5dd7070Spatrick 
725*12c85518Srobert   // Add non-standard, platform-specific search paths, e.g., for DriverKit:
726*12c85518Srobert   //  -L<sysroot>/System/DriverKit/usr/lib
727*12c85518Srobert   //  -F<sysroot>/System/DriverKit/System/Library/Framework
728*12c85518Srobert   {
729*12c85518Srobert     bool NonStandardSearchPath = false;
730*12c85518Srobert     const auto &Triple = getToolChain().getTriple();
731*12c85518Srobert     if (Triple.isDriverKit()) {
732*12c85518Srobert       // ld64 fixed the implicit -F and -L paths in ld64-605.1+.
733*12c85518Srobert       NonStandardSearchPath =
734*12c85518Srobert           Version.getMajor() < 605 ||
735*12c85518Srobert           (Version.getMajor() == 605 && Version.getMinor().value_or(0) < 1);
736*12c85518Srobert     }
737*12c85518Srobert 
738*12c85518Srobert     if (NonStandardSearchPath) {
739*12c85518Srobert       if (auto *Sysroot = Args.getLastArg(options::OPT_isysroot)) {
740*12c85518Srobert         auto AddSearchPath = [&](StringRef Flag, StringRef SearchPath) {
741*12c85518Srobert           SmallString<128> P(Sysroot->getValue());
742*12c85518Srobert           AppendPlatformPrefix(P, Triple);
743*12c85518Srobert           llvm::sys::path::append(P, SearchPath);
744*12c85518Srobert           if (getToolChain().getVFS().exists(P)) {
745*12c85518Srobert             CmdArgs.push_back(Args.MakeArgString(Flag + P));
746*12c85518Srobert           }
747*12c85518Srobert         };
748*12c85518Srobert         AddSearchPath("-L", "/usr/lib");
749*12c85518Srobert         AddSearchPath("-F", "/System/Library/Frameworks");
750*12c85518Srobert       }
751*12c85518Srobert     }
752*12c85518Srobert   }
753*12c85518Srobert 
754a9ac8606Spatrick   ResponseFileSupport ResponseSupport;
755*12c85518Srobert   if (Version >= VersionTuple(705) || LinkerIsLLD) {
756a9ac8606Spatrick     ResponseSupport = ResponseFileSupport::AtFileUTF8();
757a9ac8606Spatrick   } else {
758ec727ea7Spatrick     // For older versions of the linker, use the legacy filelist method instead.
759ec727ea7Spatrick     ResponseSupport = {ResponseFileSupport::RF_FileList, llvm::sys::WEM_UTF8,
760ec727ea7Spatrick                        "-filelist"};
761ec727ea7Spatrick   }
762ec727ea7Spatrick 
763ec727ea7Spatrick   std::unique_ptr<Command> Cmd = std::make_unique<Command>(
764a9ac8606Spatrick       JA, *this, ResponseSupport, Exec, CmdArgs, Inputs, Output);
765e5dd7070Spatrick   Cmd->setInputFileList(std::move(InputFileList));
766e5dd7070Spatrick   C.addCommand(std::move(Cmd));
767e5dd7070Spatrick }
768e5dd7070Spatrick 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const769*12c85518Srobert void darwin::StaticLibTool::ConstructJob(Compilation &C, const JobAction &JA,
770*12c85518Srobert                                          const InputInfo &Output,
771*12c85518Srobert                                          const InputInfoList &Inputs,
772*12c85518Srobert                                          const ArgList &Args,
773*12c85518Srobert                                          const char *LinkingOutput) const {
774*12c85518Srobert   const Driver &D = getToolChain().getDriver();
775*12c85518Srobert 
776*12c85518Srobert   // Silence warning for "clang -g foo.o -o foo"
777*12c85518Srobert   Args.ClaimAllArgs(options::OPT_g_Group);
778*12c85518Srobert   // and "clang -emit-llvm foo.o -o foo"
779*12c85518Srobert   Args.ClaimAllArgs(options::OPT_emit_llvm);
780*12c85518Srobert   // and for "clang -w foo.o -o foo". Other warning options are already
781*12c85518Srobert   // handled somewhere else.
782*12c85518Srobert   Args.ClaimAllArgs(options::OPT_w);
783*12c85518Srobert   // Silence warnings when linking C code with a C++ '-stdlib' argument.
784*12c85518Srobert   Args.ClaimAllArgs(options::OPT_stdlib_EQ);
785*12c85518Srobert 
786*12c85518Srobert   // libtool <options> <output_file> <input_files>
787*12c85518Srobert   ArgStringList CmdArgs;
788*12c85518Srobert   // Create and insert file members with a deterministic index.
789*12c85518Srobert   CmdArgs.push_back("-static");
790*12c85518Srobert   CmdArgs.push_back("-D");
791*12c85518Srobert   CmdArgs.push_back("-no_warning_for_no_symbols");
792*12c85518Srobert   CmdArgs.push_back("-o");
793*12c85518Srobert   CmdArgs.push_back(Output.getFilename());
794*12c85518Srobert 
795*12c85518Srobert   for (const auto &II : Inputs) {
796*12c85518Srobert     if (II.isFilename()) {
797*12c85518Srobert       CmdArgs.push_back(II.getFilename());
798*12c85518Srobert     }
799*12c85518Srobert   }
800*12c85518Srobert 
801*12c85518Srobert   // Delete old output archive file if it already exists before generating a new
802*12c85518Srobert   // archive file.
803*12c85518Srobert   const auto *OutputFileName = Output.getFilename();
804*12c85518Srobert   if (Output.isFilename() && llvm::sys::fs::exists(OutputFileName)) {
805*12c85518Srobert     if (std::error_code EC = llvm::sys::fs::remove(OutputFileName)) {
806*12c85518Srobert       D.Diag(diag::err_drv_unable_to_remove_file) << EC.message();
807*12c85518Srobert       return;
808*12c85518Srobert     }
809*12c85518Srobert   }
810*12c85518Srobert 
811*12c85518Srobert   const char *Exec = Args.MakeArgString(getToolChain().GetStaticLibToolPath());
812*12c85518Srobert   C.addCommand(std::make_unique<Command>(JA, *this,
813*12c85518Srobert                                          ResponseFileSupport::AtFileUTF8(),
814*12c85518Srobert                                          Exec, CmdArgs, Inputs, Output));
815*12c85518Srobert }
816*12c85518Srobert 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const817e5dd7070Spatrick void darwin::Lipo::ConstructJob(Compilation &C, const JobAction &JA,
818e5dd7070Spatrick                                 const InputInfo &Output,
819e5dd7070Spatrick                                 const InputInfoList &Inputs,
820e5dd7070Spatrick                                 const ArgList &Args,
821e5dd7070Spatrick                                 const char *LinkingOutput) const {
822e5dd7070Spatrick   ArgStringList CmdArgs;
823e5dd7070Spatrick 
824e5dd7070Spatrick   CmdArgs.push_back("-create");
825e5dd7070Spatrick   assert(Output.isFilename() && "Unexpected lipo output.");
826e5dd7070Spatrick 
827e5dd7070Spatrick   CmdArgs.push_back("-output");
828e5dd7070Spatrick   CmdArgs.push_back(Output.getFilename());
829e5dd7070Spatrick 
830e5dd7070Spatrick   for (const auto &II : Inputs) {
831e5dd7070Spatrick     assert(II.isFilename() && "Unexpected lipo input.");
832e5dd7070Spatrick     CmdArgs.push_back(II.getFilename());
833e5dd7070Spatrick   }
834e5dd7070Spatrick 
835e5dd7070Spatrick   const char *Exec = Args.MakeArgString(getToolChain().GetProgramPath("lipo"));
836ec727ea7Spatrick   C.addCommand(std::make_unique<Command>(JA, *this, ResponseFileSupport::None(),
837a9ac8606Spatrick                                          Exec, CmdArgs, Inputs, Output));
838e5dd7070Spatrick }
839e5dd7070Spatrick 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const840e5dd7070Spatrick void darwin::Dsymutil::ConstructJob(Compilation &C, const JobAction &JA,
841e5dd7070Spatrick                                     const InputInfo &Output,
842e5dd7070Spatrick                                     const InputInfoList &Inputs,
843e5dd7070Spatrick                                     const ArgList &Args,
844e5dd7070Spatrick                                     const char *LinkingOutput) const {
845e5dd7070Spatrick   ArgStringList CmdArgs;
846e5dd7070Spatrick 
847e5dd7070Spatrick   CmdArgs.push_back("-o");
848e5dd7070Spatrick   CmdArgs.push_back(Output.getFilename());
849e5dd7070Spatrick 
850e5dd7070Spatrick   assert(Inputs.size() == 1 && "Unable to handle multiple inputs.");
851e5dd7070Spatrick   const InputInfo &Input = Inputs[0];
852e5dd7070Spatrick   assert(Input.isFilename() && "Unexpected dsymutil input.");
853e5dd7070Spatrick   CmdArgs.push_back(Input.getFilename());
854e5dd7070Spatrick 
855e5dd7070Spatrick   const char *Exec =
856e5dd7070Spatrick       Args.MakeArgString(getToolChain().GetProgramPath("dsymutil"));
857ec727ea7Spatrick   C.addCommand(std::make_unique<Command>(JA, *this, ResponseFileSupport::None(),
858a9ac8606Spatrick                                          Exec, CmdArgs, Inputs, Output));
859e5dd7070Spatrick }
860e5dd7070Spatrick 
ConstructJob(Compilation & C,const JobAction & JA,const InputInfo & Output,const InputInfoList & Inputs,const ArgList & Args,const char * LinkingOutput) const861e5dd7070Spatrick void darwin::VerifyDebug::ConstructJob(Compilation &C, const JobAction &JA,
862e5dd7070Spatrick                                        const InputInfo &Output,
863e5dd7070Spatrick                                        const InputInfoList &Inputs,
864e5dd7070Spatrick                                        const ArgList &Args,
865e5dd7070Spatrick                                        const char *LinkingOutput) const {
866e5dd7070Spatrick   ArgStringList CmdArgs;
867e5dd7070Spatrick   CmdArgs.push_back("--verify");
868e5dd7070Spatrick   CmdArgs.push_back("--debug-info");
869e5dd7070Spatrick   CmdArgs.push_back("--eh-frame");
870e5dd7070Spatrick   CmdArgs.push_back("--quiet");
871e5dd7070Spatrick 
872e5dd7070Spatrick   assert(Inputs.size() == 1 && "Unable to handle multiple inputs.");
873e5dd7070Spatrick   const InputInfo &Input = Inputs[0];
874e5dd7070Spatrick   assert(Input.isFilename() && "Unexpected verify input");
875e5dd7070Spatrick 
876e5dd7070Spatrick   // Grabbing the output of the earlier dsymutil run.
877e5dd7070Spatrick   CmdArgs.push_back(Input.getFilename());
878e5dd7070Spatrick 
879e5dd7070Spatrick   const char *Exec =
880e5dd7070Spatrick       Args.MakeArgString(getToolChain().GetProgramPath("dwarfdump"));
881ec727ea7Spatrick   C.addCommand(std::make_unique<Command>(JA, *this, ResponseFileSupport::None(),
882a9ac8606Spatrick                                          Exec, CmdArgs, Inputs, Output));
883e5dd7070Spatrick }
884e5dd7070Spatrick 
MachO(const Driver & D,const llvm::Triple & Triple,const ArgList & Args)885e5dd7070Spatrick MachO::MachO(const Driver &D, const llvm::Triple &Triple, const ArgList &Args)
886e5dd7070Spatrick     : ToolChain(D, Triple, Args) {
887e5dd7070Spatrick   // We expect 'as', 'ld', etc. to be adjacent to our install dir.
888e5dd7070Spatrick   getProgramPaths().push_back(getDriver().getInstalledDir());
889e5dd7070Spatrick   if (getDriver().getInstalledDir() != getDriver().Dir)
890e5dd7070Spatrick     getProgramPaths().push_back(getDriver().Dir);
891e5dd7070Spatrick }
892e5dd7070Spatrick 
893e5dd7070Spatrick /// Darwin - Darwin tool chain for i386 and x86_64.
Darwin(const Driver & D,const llvm::Triple & Triple,const ArgList & Args)894e5dd7070Spatrick Darwin::Darwin(const Driver &D, const llvm::Triple &Triple, const ArgList &Args)
895e5dd7070Spatrick     : MachO(D, Triple, Args), TargetInitialized(false),
896ec727ea7Spatrick       CudaInstallation(D, Triple, Args), RocmInstallation(D, Triple, Args) {}
897e5dd7070Spatrick 
LookupTypeForExtension(StringRef Ext) const898e5dd7070Spatrick types::ID MachO::LookupTypeForExtension(StringRef Ext) const {
899e5dd7070Spatrick   types::ID Ty = ToolChain::LookupTypeForExtension(Ext);
900e5dd7070Spatrick 
901e5dd7070Spatrick   // Darwin always preprocesses assembly files (unless -x is used explicitly).
902e5dd7070Spatrick   if (Ty == types::TY_PP_Asm)
903e5dd7070Spatrick     return types::TY_Asm;
904e5dd7070Spatrick 
905e5dd7070Spatrick   return Ty;
906e5dd7070Spatrick }
907e5dd7070Spatrick 
HasNativeLLVMSupport() const908e5dd7070Spatrick bool MachO::HasNativeLLVMSupport() const { return true; }
909e5dd7070Spatrick 
GetDefaultCXXStdlibType() const910e5dd7070Spatrick ToolChain::CXXStdlibType Darwin::GetDefaultCXXStdlibType() const {
911*12c85518Srobert   // Always use libc++ by default
912e5dd7070Spatrick   return ToolChain::CST_Libcxx;
913e5dd7070Spatrick }
914e5dd7070Spatrick 
915e5dd7070Spatrick /// Darwin provides an ARC runtime starting in MacOS X 10.7 and iOS 5.0.
getDefaultObjCRuntime(bool isNonFragile) const916e5dd7070Spatrick ObjCRuntime Darwin::getDefaultObjCRuntime(bool isNonFragile) const {
917e5dd7070Spatrick   if (isTargetWatchOSBased())
918e5dd7070Spatrick     return ObjCRuntime(ObjCRuntime::WatchOS, TargetVersion);
919e5dd7070Spatrick   if (isTargetIOSBased())
920e5dd7070Spatrick     return ObjCRuntime(ObjCRuntime::iOS, TargetVersion);
921e5dd7070Spatrick   if (isNonFragile)
922e5dd7070Spatrick     return ObjCRuntime(ObjCRuntime::MacOSX, TargetVersion);
923e5dd7070Spatrick   return ObjCRuntime(ObjCRuntime::FragileMacOSX, TargetVersion);
924e5dd7070Spatrick }
925e5dd7070Spatrick 
926e5dd7070Spatrick /// Darwin provides a blocks runtime starting in MacOS X 10.6 and iOS 3.2.
hasBlocksRuntime() const927e5dd7070Spatrick bool Darwin::hasBlocksRuntime() const {
928*12c85518Srobert   if (isTargetWatchOSBased() || isTargetDriverKit())
929e5dd7070Spatrick     return true;
930e5dd7070Spatrick   else if (isTargetIOSBased())
931e5dd7070Spatrick     return !isIPhoneOSVersionLT(3, 2);
932e5dd7070Spatrick   else {
933a9ac8606Spatrick     assert(isTargetMacOSBased() && "unexpected darwin target");
934e5dd7070Spatrick     return !isMacosxVersionLT(10, 6);
935e5dd7070Spatrick   }
936e5dd7070Spatrick }
937e5dd7070Spatrick 
AddCudaIncludeArgs(const ArgList & DriverArgs,ArgStringList & CC1Args) const938e5dd7070Spatrick void Darwin::AddCudaIncludeArgs(const ArgList &DriverArgs,
939e5dd7070Spatrick                                 ArgStringList &CC1Args) const {
940e5dd7070Spatrick   CudaInstallation.AddCudaIncludeArgs(DriverArgs, CC1Args);
941e5dd7070Spatrick }
942e5dd7070Spatrick 
AddHIPIncludeArgs(const ArgList & DriverArgs,ArgStringList & CC1Args) const943ec727ea7Spatrick void Darwin::AddHIPIncludeArgs(const ArgList &DriverArgs,
944ec727ea7Spatrick                                ArgStringList &CC1Args) const {
945ec727ea7Spatrick   RocmInstallation.AddHIPIncludeArgs(DriverArgs, CC1Args);
946ec727ea7Spatrick }
947ec727ea7Spatrick 
948e5dd7070Spatrick // This is just a MachO name translation routine and there's no
949e5dd7070Spatrick // way to join this into ARMTargetParser without breaking all
950e5dd7070Spatrick // other assumptions. Maybe MachO should consider standardising
951e5dd7070Spatrick // their nomenclature.
ArmMachOArchName(StringRef Arch)952e5dd7070Spatrick static const char *ArmMachOArchName(StringRef Arch) {
953e5dd7070Spatrick   return llvm::StringSwitch<const char *>(Arch)
954e5dd7070Spatrick       .Case("armv6k", "armv6")
955e5dd7070Spatrick       .Case("armv6m", "armv6m")
956e5dd7070Spatrick       .Case("armv5tej", "armv5")
957e5dd7070Spatrick       .Case("xscale", "xscale")
958e5dd7070Spatrick       .Case("armv4t", "armv4t")
959e5dd7070Spatrick       .Case("armv7", "armv7")
960e5dd7070Spatrick       .Cases("armv7a", "armv7-a", "armv7")
961e5dd7070Spatrick       .Cases("armv7r", "armv7-r", "armv7")
962e5dd7070Spatrick       .Cases("armv7em", "armv7e-m", "armv7em")
963e5dd7070Spatrick       .Cases("armv7k", "armv7-k", "armv7k")
964e5dd7070Spatrick       .Cases("armv7m", "armv7-m", "armv7m")
965e5dd7070Spatrick       .Cases("armv7s", "armv7-s", "armv7s")
966e5dd7070Spatrick       .Default(nullptr);
967e5dd7070Spatrick }
968e5dd7070Spatrick 
ArmMachOArchNameCPU(StringRef CPU)969e5dd7070Spatrick static const char *ArmMachOArchNameCPU(StringRef CPU) {
970e5dd7070Spatrick   llvm::ARM::ArchKind ArchKind = llvm::ARM::parseCPUArch(CPU);
971e5dd7070Spatrick   if (ArchKind == llvm::ARM::ArchKind::INVALID)
972e5dd7070Spatrick     return nullptr;
973e5dd7070Spatrick   StringRef Arch = llvm::ARM::getArchName(ArchKind);
974e5dd7070Spatrick 
975e5dd7070Spatrick   // FIXME: Make sure this MachO triple mangling is really necessary.
976e5dd7070Spatrick   // ARMv5* normalises to ARMv5.
977e5dd7070Spatrick   if (Arch.startswith("armv5"))
978e5dd7070Spatrick     Arch = Arch.substr(0, 5);
979e5dd7070Spatrick   // ARMv6*, except ARMv6M, normalises to ARMv6.
980e5dd7070Spatrick   else if (Arch.startswith("armv6") && !Arch.endswith("6m"))
981e5dd7070Spatrick     Arch = Arch.substr(0, 5);
982e5dd7070Spatrick   // ARMv7A normalises to ARMv7.
983e5dd7070Spatrick   else if (Arch.endswith("v7a"))
984e5dd7070Spatrick     Arch = Arch.substr(0, 5);
985e5dd7070Spatrick   return Arch.data();
986e5dd7070Spatrick }
987e5dd7070Spatrick 
getMachOArchName(const ArgList & Args) const988e5dd7070Spatrick StringRef MachO::getMachOArchName(const ArgList &Args) const {
989e5dd7070Spatrick   switch (getTriple().getArch()) {
990e5dd7070Spatrick   default:
991e5dd7070Spatrick     return getDefaultUniversalArchName();
992e5dd7070Spatrick 
993e5dd7070Spatrick   case llvm::Triple::aarch64_32:
994e5dd7070Spatrick     return "arm64_32";
995e5dd7070Spatrick 
996a9ac8606Spatrick   case llvm::Triple::aarch64: {
997a9ac8606Spatrick     if (getTriple().isArm64e())
998a9ac8606Spatrick       return "arm64e";
999e5dd7070Spatrick     return "arm64";
1000a9ac8606Spatrick   }
1001e5dd7070Spatrick 
1002e5dd7070Spatrick   case llvm::Triple::thumb:
1003e5dd7070Spatrick   case llvm::Triple::arm:
1004e5dd7070Spatrick     if (const Arg *A = Args.getLastArg(clang::driver::options::OPT_march_EQ))
1005e5dd7070Spatrick       if (const char *Arch = ArmMachOArchName(A->getValue()))
1006e5dd7070Spatrick         return Arch;
1007e5dd7070Spatrick 
1008e5dd7070Spatrick     if (const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ))
1009e5dd7070Spatrick       if (const char *Arch = ArmMachOArchNameCPU(A->getValue()))
1010e5dd7070Spatrick         return Arch;
1011e5dd7070Spatrick 
1012e5dd7070Spatrick     return "arm";
1013e5dd7070Spatrick   }
1014e5dd7070Spatrick }
1015e5dd7070Spatrick 
getLinkerVersion(const llvm::opt::ArgList & Args) const1016*12c85518Srobert VersionTuple MachO::getLinkerVersion(const llvm::opt::ArgList &Args) const {
1017*12c85518Srobert   if (LinkerVersion) {
1018*12c85518Srobert #ifndef NDEBUG
1019*12c85518Srobert     VersionTuple NewLinkerVersion;
1020*12c85518Srobert     if (Arg *A = Args.getLastArg(options::OPT_mlinker_version_EQ))
1021*12c85518Srobert       (void)NewLinkerVersion.tryParse(A->getValue());
1022*12c85518Srobert     assert(NewLinkerVersion == LinkerVersion);
1023*12c85518Srobert #endif
1024*12c85518Srobert     return *LinkerVersion;
1025*12c85518Srobert   }
1026*12c85518Srobert 
1027*12c85518Srobert   VersionTuple NewLinkerVersion;
1028*12c85518Srobert   if (Arg *A = Args.getLastArg(options::OPT_mlinker_version_EQ))
1029*12c85518Srobert     if (NewLinkerVersion.tryParse(A->getValue()))
1030*12c85518Srobert       getDriver().Diag(diag::err_drv_invalid_version_number)
1031*12c85518Srobert         << A->getAsString(Args);
1032*12c85518Srobert 
1033*12c85518Srobert   LinkerVersion = NewLinkerVersion;
1034*12c85518Srobert   return *LinkerVersion;
1035*12c85518Srobert }
1036*12c85518Srobert 
~Darwin()1037e5dd7070Spatrick Darwin::~Darwin() {}
1038e5dd7070Spatrick 
~MachO()1039e5dd7070Spatrick MachO::~MachO() {}
1040e5dd7070Spatrick 
ComputeEffectiveClangTriple(const ArgList & Args,types::ID InputType) const1041e5dd7070Spatrick std::string Darwin::ComputeEffectiveClangTriple(const ArgList &Args,
1042e5dd7070Spatrick                                                 types::ID InputType) const {
1043e5dd7070Spatrick   llvm::Triple Triple(ComputeLLVMTriple(Args, InputType));
1044e5dd7070Spatrick 
1045e5dd7070Spatrick   // If the target isn't initialized (e.g., an unknown Darwin platform, return
1046e5dd7070Spatrick   // the default triple).
1047e5dd7070Spatrick   if (!isTargetInitialized())
1048e5dd7070Spatrick     return Triple.getTriple();
1049e5dd7070Spatrick 
1050e5dd7070Spatrick   SmallString<16> Str;
1051e5dd7070Spatrick   if (isTargetWatchOSBased())
1052e5dd7070Spatrick     Str += "watchos";
1053e5dd7070Spatrick   else if (isTargetTvOSBased())
1054e5dd7070Spatrick     Str += "tvos";
1055*12c85518Srobert   else if (isTargetDriverKit())
1056*12c85518Srobert     Str += "driverkit";
1057a9ac8606Spatrick   else if (isTargetIOSBased() || isTargetMacCatalyst())
1058e5dd7070Spatrick     Str += "ios";
1059e5dd7070Spatrick   else
1060e5dd7070Spatrick     Str += "macosx";
1061a9ac8606Spatrick   Str += getTripleTargetVersion().getAsString();
1062e5dd7070Spatrick   Triple.setOSName(Str);
1063e5dd7070Spatrick 
1064e5dd7070Spatrick   return Triple.getTriple();
1065e5dd7070Spatrick }
1066e5dd7070Spatrick 
getTool(Action::ActionClass AC) const1067e5dd7070Spatrick Tool *MachO::getTool(Action::ActionClass AC) const {
1068e5dd7070Spatrick   switch (AC) {
1069e5dd7070Spatrick   case Action::LipoJobClass:
1070e5dd7070Spatrick     if (!Lipo)
1071e5dd7070Spatrick       Lipo.reset(new tools::darwin::Lipo(*this));
1072e5dd7070Spatrick     return Lipo.get();
1073e5dd7070Spatrick   case Action::DsymutilJobClass:
1074e5dd7070Spatrick     if (!Dsymutil)
1075e5dd7070Spatrick       Dsymutil.reset(new tools::darwin::Dsymutil(*this));
1076e5dd7070Spatrick     return Dsymutil.get();
1077e5dd7070Spatrick   case Action::VerifyDebugInfoJobClass:
1078e5dd7070Spatrick     if (!VerifyDebug)
1079e5dd7070Spatrick       VerifyDebug.reset(new tools::darwin::VerifyDebug(*this));
1080e5dd7070Spatrick     return VerifyDebug.get();
1081e5dd7070Spatrick   default:
1082e5dd7070Spatrick     return ToolChain::getTool(AC);
1083e5dd7070Spatrick   }
1084e5dd7070Spatrick }
1085e5dd7070Spatrick 
buildLinker() const1086e5dd7070Spatrick Tool *MachO::buildLinker() const { return new tools::darwin::Linker(*this); }
1087e5dd7070Spatrick 
buildStaticLibTool() const1088*12c85518Srobert Tool *MachO::buildStaticLibTool() const {
1089*12c85518Srobert   return new tools::darwin::StaticLibTool(*this);
1090*12c85518Srobert }
1091*12c85518Srobert 
buildAssembler() const1092e5dd7070Spatrick Tool *MachO::buildAssembler() const {
1093e5dd7070Spatrick   return new tools::darwin::Assembler(*this);
1094e5dd7070Spatrick }
1095e5dd7070Spatrick 
DarwinClang(const Driver & D,const llvm::Triple & Triple,const ArgList & Args)1096e5dd7070Spatrick DarwinClang::DarwinClang(const Driver &D, const llvm::Triple &Triple,
1097e5dd7070Spatrick                          const ArgList &Args)
1098e5dd7070Spatrick     : Darwin(D, Triple, Args) {}
1099e5dd7070Spatrick 
addClangWarningOptions(ArgStringList & CC1Args) const1100e5dd7070Spatrick void DarwinClang::addClangWarningOptions(ArgStringList &CC1Args) const {
1101ec727ea7Spatrick   // Always error about undefined 'TARGET_OS_*' macros.
1102ec727ea7Spatrick   CC1Args.push_back("-Wundef-prefix=TARGET_OS_");
1103ec727ea7Spatrick   CC1Args.push_back("-Werror=undef-prefix");
1104ec727ea7Spatrick 
1105e5dd7070Spatrick   // For modern targets, promote certain warnings to errors.
1106e5dd7070Spatrick   if (isTargetWatchOSBased() || getTriple().isArch64Bit()) {
1107e5dd7070Spatrick     // Always enable -Wdeprecated-objc-isa-usage and promote it
1108e5dd7070Spatrick     // to an error.
1109e5dd7070Spatrick     CC1Args.push_back("-Wdeprecated-objc-isa-usage");
1110e5dd7070Spatrick     CC1Args.push_back("-Werror=deprecated-objc-isa-usage");
1111e5dd7070Spatrick 
1112e5dd7070Spatrick     // For iOS and watchOS, also error about implicit function declarations,
1113e5dd7070Spatrick     // as that can impact calling conventions.
1114e5dd7070Spatrick     if (!isTargetMacOS())
1115e5dd7070Spatrick       CC1Args.push_back("-Werror=implicit-function-declaration");
1116e5dd7070Spatrick   }
1117e5dd7070Spatrick }
1118e5dd7070Spatrick 
1119e5dd7070Spatrick /// Take a path that speculatively points into Xcode and return the
1120e5dd7070Spatrick /// `XCODE/Contents/Developer` path if it is an Xcode path, or an empty path
1121e5dd7070Spatrick /// otherwise.
getXcodeDeveloperPath(StringRef PathIntoXcode)1122e5dd7070Spatrick static StringRef getXcodeDeveloperPath(StringRef PathIntoXcode) {
1123e5dd7070Spatrick   static constexpr llvm::StringLiteral XcodeAppSuffix(
1124e5dd7070Spatrick       ".app/Contents/Developer");
1125e5dd7070Spatrick   size_t Index = PathIntoXcode.find(XcodeAppSuffix);
1126e5dd7070Spatrick   if (Index == StringRef::npos)
1127e5dd7070Spatrick     return "";
1128e5dd7070Spatrick   return PathIntoXcode.take_front(Index + XcodeAppSuffix.size());
1129e5dd7070Spatrick }
1130e5dd7070Spatrick 
AddLinkARCArgs(const ArgList & Args,ArgStringList & CmdArgs) const1131e5dd7070Spatrick void DarwinClang::AddLinkARCArgs(const ArgList &Args,
1132e5dd7070Spatrick                                  ArgStringList &CmdArgs) const {
1133e5dd7070Spatrick   // Avoid linking compatibility stubs on i386 mac.
1134a9ac8606Spatrick   if (isTargetMacOSBased() && getArch() == llvm::Triple::x86)
1135e5dd7070Spatrick     return;
1136ec727ea7Spatrick   if (isTargetAppleSiliconMac())
1137ec727ea7Spatrick     return;
1138a9ac8606Spatrick   // ARC runtime is supported everywhere on arm64e.
1139a9ac8606Spatrick   if (getTriple().isArm64e())
1140a9ac8606Spatrick     return;
1141e5dd7070Spatrick 
1142e5dd7070Spatrick   ObjCRuntime runtime = getDefaultObjCRuntime(/*nonfragile*/ true);
1143e5dd7070Spatrick 
1144e5dd7070Spatrick   if ((runtime.hasNativeARC() || !isObjCAutoRefCount(Args)) &&
1145e5dd7070Spatrick       runtime.hasSubscripting())
1146e5dd7070Spatrick     return;
1147e5dd7070Spatrick 
1148e5dd7070Spatrick   SmallString<128> P(getDriver().ClangExecutable);
1149e5dd7070Spatrick   llvm::sys::path::remove_filename(P); // 'clang'
1150e5dd7070Spatrick   llvm::sys::path::remove_filename(P); // 'bin'
1151*12c85518Srobert   llvm::sys::path::append(P, "lib", "arc");
1152e5dd7070Spatrick 
1153e5dd7070Spatrick   // 'libarclite' usually lives in the same toolchain as 'clang'. However, the
1154e5dd7070Spatrick   // Swift open source toolchains for macOS distribute Clang without libarclite.
1155e5dd7070Spatrick   // In that case, to allow the linker to find 'libarclite', we point to the
1156e5dd7070Spatrick   // 'libarclite' in the XcodeDefault toolchain instead.
1157*12c85518Srobert   if (!getVFS().exists(P)) {
1158*12c85518Srobert     auto updatePath = [&](const Arg *A) {
1159e5dd7070Spatrick       // Try to infer the path to 'libarclite' in the toolchain from the
1160e5dd7070Spatrick       // specified SDK path.
1161e5dd7070Spatrick       StringRef XcodePathForSDK = getXcodeDeveloperPath(A->getValue());
1162*12c85518Srobert       if (XcodePathForSDK.empty())
1163*12c85518Srobert         return false;
1164*12c85518Srobert 
1165e5dd7070Spatrick       P = XcodePathForSDK;
1166*12c85518Srobert       llvm::sys::path::append(P, "Toolchains/XcodeDefault.xctoolchain/usr",
1167*12c85518Srobert                               "lib", "arc");
1168*12c85518Srobert       return getVFS().exists(P);
1169*12c85518Srobert     };
1170*12c85518Srobert 
1171*12c85518Srobert     bool updated = false;
1172*12c85518Srobert     if (const Arg *A = Args.getLastArg(options::OPT_isysroot))
1173*12c85518Srobert       updated = updatePath(A);
1174*12c85518Srobert 
1175*12c85518Srobert     if (!updated) {
1176*12c85518Srobert       if (const Arg *A = Args.getLastArg(options::OPT__sysroot_EQ))
1177*12c85518Srobert         updatePath(A);
1178e5dd7070Spatrick     }
1179e5dd7070Spatrick   }
1180e5dd7070Spatrick 
1181e5dd7070Spatrick   CmdArgs.push_back("-force_load");
1182*12c85518Srobert   llvm::sys::path::append(P, "libarclite_");
1183e5dd7070Spatrick   // Mash in the platform.
1184e5dd7070Spatrick   if (isTargetWatchOSSimulator())
1185e5dd7070Spatrick     P += "watchsimulator";
1186e5dd7070Spatrick   else if (isTargetWatchOS())
1187e5dd7070Spatrick     P += "watchos";
1188e5dd7070Spatrick   else if (isTargetTvOSSimulator())
1189e5dd7070Spatrick     P += "appletvsimulator";
1190e5dd7070Spatrick   else if (isTargetTvOS())
1191e5dd7070Spatrick     P += "appletvos";
1192e5dd7070Spatrick   else if (isTargetIOSSimulator())
1193e5dd7070Spatrick     P += "iphonesimulator";
1194e5dd7070Spatrick   else if (isTargetIPhoneOS())
1195e5dd7070Spatrick     P += "iphoneos";
1196e5dd7070Spatrick   else
1197e5dd7070Spatrick     P += "macosx";
1198e5dd7070Spatrick   P += ".a";
1199e5dd7070Spatrick 
1200e5dd7070Spatrick   CmdArgs.push_back(Args.MakeArgString(P));
1201e5dd7070Spatrick }
1202e5dd7070Spatrick 
GetDefaultDwarfVersion() const1203e5dd7070Spatrick unsigned DarwinClang::GetDefaultDwarfVersion() const {
1204e5dd7070Spatrick   // Default to use DWARF 2 on OS X 10.10 / iOS 8 and lower.
1205a9ac8606Spatrick   if ((isTargetMacOSBased() && isMacosxVersionLT(10, 11)) ||
1206e5dd7070Spatrick       (isTargetIOSBased() && isIPhoneOSVersionLT(9)))
1207e5dd7070Spatrick     return 2;
1208e5dd7070Spatrick   return 4;
1209e5dd7070Spatrick }
1210e5dd7070Spatrick 
AddLinkRuntimeLib(const ArgList & Args,ArgStringList & CmdArgs,StringRef Component,RuntimeLinkOptions Opts,bool IsShared) const1211e5dd7070Spatrick void MachO::AddLinkRuntimeLib(const ArgList &Args, ArgStringList &CmdArgs,
1212e5dd7070Spatrick                               StringRef Component, RuntimeLinkOptions Opts,
1213e5dd7070Spatrick                               bool IsShared) const {
1214e5dd7070Spatrick   SmallString<64> DarwinLibName = StringRef("libclang_rt.");
1215e5dd7070Spatrick   // an Darwin the builtins compomnent is not in the library name
1216e5dd7070Spatrick   if (Component != "builtins") {
1217e5dd7070Spatrick     DarwinLibName += Component;
1218e5dd7070Spatrick     if (!(Opts & RLO_IsEmbedded))
1219e5dd7070Spatrick       DarwinLibName += "_";
1220a9ac8606Spatrick   }
1221e5dd7070Spatrick 
1222a9ac8606Spatrick   DarwinLibName += getOSLibraryNameSuffix();
1223e5dd7070Spatrick   DarwinLibName += IsShared ? "_dynamic.dylib" : ".a";
1224e5dd7070Spatrick   SmallString<128> Dir(getDriver().ResourceDir);
1225*12c85518Srobert   llvm::sys::path::append(Dir, "lib", "darwin");
1226*12c85518Srobert   if (Opts & RLO_IsEmbedded)
1227*12c85518Srobert     llvm::sys::path::append(Dir, "macho_embedded");
1228e5dd7070Spatrick 
1229e5dd7070Spatrick   SmallString<128> P(Dir);
1230e5dd7070Spatrick   llvm::sys::path::append(P, DarwinLibName);
1231e5dd7070Spatrick 
1232e5dd7070Spatrick   // For now, allow missing resource libraries to support developers who may
1233e5dd7070Spatrick   // not have compiler-rt checked out or integrated into their build (unless
1234e5dd7070Spatrick   // we explicitly force linking with this library).
1235e5dd7070Spatrick   if ((Opts & RLO_AlwaysLink) || getVFS().exists(P)) {
1236e5dd7070Spatrick     const char *LibArg = Args.MakeArgString(P);
1237e5dd7070Spatrick     CmdArgs.push_back(LibArg);
1238e5dd7070Spatrick   }
1239e5dd7070Spatrick 
1240e5dd7070Spatrick   // Adding the rpaths might negatively interact when other rpaths are involved,
1241e5dd7070Spatrick   // so we should make sure we add the rpaths last, after all user-specified
1242e5dd7070Spatrick   // rpaths. This is currently true from this place, but we need to be
1243e5dd7070Spatrick   // careful if this function is ever called before user's rpaths are emitted.
1244e5dd7070Spatrick   if (Opts & RLO_AddRPath) {
1245e5dd7070Spatrick     assert(DarwinLibName.endswith(".dylib") && "must be a dynamic library");
1246e5dd7070Spatrick 
1247e5dd7070Spatrick     // Add @executable_path to rpath to support having the dylib copied with
1248e5dd7070Spatrick     // the executable.
1249e5dd7070Spatrick     CmdArgs.push_back("-rpath");
1250e5dd7070Spatrick     CmdArgs.push_back("@executable_path");
1251e5dd7070Spatrick 
1252e5dd7070Spatrick     // Add the path to the resource dir to rpath to support using the dylib
1253e5dd7070Spatrick     // from the default location without copying.
1254e5dd7070Spatrick     CmdArgs.push_back("-rpath");
1255e5dd7070Spatrick     CmdArgs.push_back(Args.MakeArgString(Dir));
1256e5dd7070Spatrick   }
1257e5dd7070Spatrick }
1258e5dd7070Spatrick 
getPlatformFamily() const1259e5dd7070Spatrick StringRef Darwin::getPlatformFamily() const {
1260e5dd7070Spatrick   switch (TargetPlatform) {
1261e5dd7070Spatrick     case DarwinPlatformKind::MacOS:
1262e5dd7070Spatrick       return "MacOSX";
1263e5dd7070Spatrick     case DarwinPlatformKind::IPhoneOS:
1264a9ac8606Spatrick       if (TargetEnvironment == MacCatalyst)
1265a9ac8606Spatrick         return "MacOSX";
1266e5dd7070Spatrick       return "iPhone";
1267e5dd7070Spatrick     case DarwinPlatformKind::TvOS:
1268e5dd7070Spatrick       return "AppleTV";
1269e5dd7070Spatrick     case DarwinPlatformKind::WatchOS:
1270e5dd7070Spatrick       return "Watch";
1271*12c85518Srobert     case DarwinPlatformKind::DriverKit:
1272*12c85518Srobert       return "DriverKit";
1273e5dd7070Spatrick   }
1274e5dd7070Spatrick   llvm_unreachable("Unsupported platform");
1275e5dd7070Spatrick }
1276e5dd7070Spatrick 
getSDKName(StringRef isysroot)1277e5dd7070Spatrick StringRef Darwin::getSDKName(StringRef isysroot) {
1278e5dd7070Spatrick   // Assume SDK has path: SOME_PATH/SDKs/PlatformXX.YY.sdk
1279ec727ea7Spatrick   auto BeginSDK = llvm::sys::path::rbegin(isysroot);
1280ec727ea7Spatrick   auto EndSDK = llvm::sys::path::rend(isysroot);
1281e5dd7070Spatrick   for (auto IT = BeginSDK; IT != EndSDK; ++IT) {
1282e5dd7070Spatrick     StringRef SDK = *IT;
1283e5dd7070Spatrick     if (SDK.endswith(".sdk"))
1284e5dd7070Spatrick       return SDK.slice(0, SDK.size() - 4);
1285e5dd7070Spatrick   }
1286e5dd7070Spatrick   return "";
1287e5dd7070Spatrick }
1288e5dd7070Spatrick 
getOSLibraryNameSuffix(bool IgnoreSim) const1289e5dd7070Spatrick StringRef Darwin::getOSLibraryNameSuffix(bool IgnoreSim) const {
1290e5dd7070Spatrick   switch (TargetPlatform) {
1291e5dd7070Spatrick   case DarwinPlatformKind::MacOS:
1292e5dd7070Spatrick     return "osx";
1293e5dd7070Spatrick   case DarwinPlatformKind::IPhoneOS:
1294a9ac8606Spatrick     if (TargetEnvironment == MacCatalyst)
1295a9ac8606Spatrick       return "osx";
1296e5dd7070Spatrick     return TargetEnvironment == NativeEnvironment || IgnoreSim ? "ios"
1297e5dd7070Spatrick                                                                : "iossim";
1298e5dd7070Spatrick   case DarwinPlatformKind::TvOS:
1299e5dd7070Spatrick     return TargetEnvironment == NativeEnvironment || IgnoreSim ? "tvos"
1300e5dd7070Spatrick                                                                : "tvossim";
1301e5dd7070Spatrick   case DarwinPlatformKind::WatchOS:
1302e5dd7070Spatrick     return TargetEnvironment == NativeEnvironment || IgnoreSim ? "watchos"
1303e5dd7070Spatrick                                                                : "watchossim";
1304*12c85518Srobert   case DarwinPlatformKind::DriverKit:
1305*12c85518Srobert     return "driverkit";
1306e5dd7070Spatrick   }
1307e5dd7070Spatrick   llvm_unreachable("Unsupported platform");
1308e5dd7070Spatrick }
1309e5dd7070Spatrick 
1310e5dd7070Spatrick /// Check if the link command contains a symbol export directive.
hasExportSymbolDirective(const ArgList & Args)1311e5dd7070Spatrick static bool hasExportSymbolDirective(const ArgList &Args) {
1312e5dd7070Spatrick   for (Arg *A : Args) {
1313e5dd7070Spatrick     if (A->getOption().matches(options::OPT_exported__symbols__list))
1314e5dd7070Spatrick       return true;
1315e5dd7070Spatrick     if (!A->getOption().matches(options::OPT_Wl_COMMA) &&
1316e5dd7070Spatrick         !A->getOption().matches(options::OPT_Xlinker))
1317e5dd7070Spatrick       continue;
1318e5dd7070Spatrick     if (A->containsValue("-exported_symbols_list") ||
1319e5dd7070Spatrick         A->containsValue("-exported_symbol"))
1320e5dd7070Spatrick       return true;
1321e5dd7070Spatrick   }
1322e5dd7070Spatrick   return false;
1323e5dd7070Spatrick }
1324e5dd7070Spatrick 
1325e5dd7070Spatrick /// Add an export directive for \p Symbol to the link command.
addExportedSymbol(ArgStringList & CmdArgs,const char * Symbol)1326e5dd7070Spatrick static void addExportedSymbol(ArgStringList &CmdArgs, const char *Symbol) {
1327e5dd7070Spatrick   CmdArgs.push_back("-exported_symbol");
1328e5dd7070Spatrick   CmdArgs.push_back(Symbol);
1329e5dd7070Spatrick }
1330e5dd7070Spatrick 
1331e5dd7070Spatrick /// Add a sectalign directive for \p Segment and \p Section to the maximum
1332e5dd7070Spatrick /// expected page size for Darwin.
1333e5dd7070Spatrick ///
1334e5dd7070Spatrick /// On iPhone 6+ the max supported page size is 16K. On macOS, the max is 4K.
1335e5dd7070Spatrick /// Use a common alignment constant (16K) for now, and reduce the alignment on
1336e5dd7070Spatrick /// macOS if it proves important.
addSectalignToPage(const ArgList & Args,ArgStringList & CmdArgs,StringRef Segment,StringRef Section)1337e5dd7070Spatrick static void addSectalignToPage(const ArgList &Args, ArgStringList &CmdArgs,
1338e5dd7070Spatrick                                StringRef Segment, StringRef Section) {
1339e5dd7070Spatrick   for (const char *A : {"-sectalign", Args.MakeArgString(Segment),
1340e5dd7070Spatrick                         Args.MakeArgString(Section), "0x4000"})
1341e5dd7070Spatrick     CmdArgs.push_back(A);
1342e5dd7070Spatrick }
1343e5dd7070Spatrick 
addProfileRTLibs(const ArgList & Args,ArgStringList & CmdArgs) const1344e5dd7070Spatrick void Darwin::addProfileRTLibs(const ArgList &Args,
1345e5dd7070Spatrick                               ArgStringList &CmdArgs) const {
1346ec727ea7Spatrick   if (!needsProfileRT(Args) && !needsGCovInstrumentation(Args))
1347ec727ea7Spatrick     return;
1348e5dd7070Spatrick 
1349e5dd7070Spatrick   AddLinkRuntimeLib(Args, CmdArgs, "profile",
1350a9ac8606Spatrick                     RuntimeLinkOptions(RLO_AlwaysLink));
1351e5dd7070Spatrick 
1352e5dd7070Spatrick   bool ForGCOV = needsGCovInstrumentation(Args);
1353e5dd7070Spatrick 
1354e5dd7070Spatrick   // If we have a symbol export directive and we're linking in the profile
1355e5dd7070Spatrick   // runtime, automatically export symbols necessary to implement some of the
1356e5dd7070Spatrick   // runtime's functionality.
1357*12c85518Srobert   if (hasExportSymbolDirective(Args) && ForGCOV) {
1358a9ac8606Spatrick     addExportedSymbol(CmdArgs, "___gcov_dump");
1359a9ac8606Spatrick     addExportedSymbol(CmdArgs, "___gcov_reset");
1360e5dd7070Spatrick     addExportedSymbol(CmdArgs, "_writeout_fn_list");
1361389bb291Spatrick     addExportedSymbol(CmdArgs, "_reset_fn_list");
1362e5dd7070Spatrick   }
1363e5dd7070Spatrick 
1364e5dd7070Spatrick   // Align __llvm_prf_{cnts,data} sections to the maximum expected page
1365e5dd7070Spatrick   // alignment. This allows profile counters to be mmap()'d to disk. Note that
1366e5dd7070Spatrick   // it's not enough to just page-align __llvm_prf_cnts: the following section
1367e5dd7070Spatrick   // must also be page-aligned so that its data is not clobbered by mmap().
1368e5dd7070Spatrick   //
1369e5dd7070Spatrick   // The section alignment is only needed when continuous profile sync is
1370e5dd7070Spatrick   // enabled, but this is expected to be the default in Xcode. Specifying the
1371e5dd7070Spatrick   // extra alignment also allows the same binary to be used with/without sync
1372e5dd7070Spatrick   // enabled.
1373e5dd7070Spatrick   if (!ForGCOV) {
1374e5dd7070Spatrick     for (auto IPSK : {llvm::IPSK_cnts, llvm::IPSK_data}) {
1375e5dd7070Spatrick       addSectalignToPage(
1376e5dd7070Spatrick           Args, CmdArgs, "__DATA",
1377e5dd7070Spatrick           llvm::getInstrProfSectionName(IPSK, llvm::Triple::MachO,
1378e5dd7070Spatrick                                         /*AddSegmentInfo=*/false));
1379e5dd7070Spatrick     }
1380e5dd7070Spatrick   }
1381e5dd7070Spatrick }
1382e5dd7070Spatrick 
AddLinkSanitizerLibArgs(const ArgList & Args,ArgStringList & CmdArgs,StringRef Sanitizer,bool Shared) const1383e5dd7070Spatrick void DarwinClang::AddLinkSanitizerLibArgs(const ArgList &Args,
1384e5dd7070Spatrick                                           ArgStringList &CmdArgs,
1385e5dd7070Spatrick                                           StringRef Sanitizer,
1386e5dd7070Spatrick                                           bool Shared) const {
1387e5dd7070Spatrick   auto RLO = RuntimeLinkOptions(RLO_AlwaysLink | (Shared ? RLO_AddRPath : 0U));
1388e5dd7070Spatrick   AddLinkRuntimeLib(Args, CmdArgs, Sanitizer, RLO, Shared);
1389e5dd7070Spatrick }
1390e5dd7070Spatrick 
GetRuntimeLibType(const ArgList & Args) const1391e5dd7070Spatrick ToolChain::RuntimeLibType DarwinClang::GetRuntimeLibType(
1392e5dd7070Spatrick     const ArgList &Args) const {
1393e5dd7070Spatrick   if (Arg* A = Args.getLastArg(options::OPT_rtlib_EQ)) {
1394e5dd7070Spatrick     StringRef Value = A->getValue();
1395*12c85518Srobert     if (Value != "compiler-rt" && Value != "platform")
1396e5dd7070Spatrick       getDriver().Diag(clang::diag::err_drv_unsupported_rtlib_for_platform)
1397e5dd7070Spatrick           << Value << "darwin";
1398e5dd7070Spatrick   }
1399e5dd7070Spatrick 
1400e5dd7070Spatrick   return ToolChain::RLT_CompilerRT;
1401e5dd7070Spatrick }
1402e5dd7070Spatrick 
AddLinkRuntimeLibArgs(const ArgList & Args,ArgStringList & CmdArgs,bool ForceLinkBuiltinRT) const1403e5dd7070Spatrick void DarwinClang::AddLinkRuntimeLibArgs(const ArgList &Args,
1404e5dd7070Spatrick                                         ArgStringList &CmdArgs,
1405e5dd7070Spatrick                                         bool ForceLinkBuiltinRT) const {
1406e5dd7070Spatrick   // Call once to ensure diagnostic is printed if wrong value was specified
1407e5dd7070Spatrick   GetRuntimeLibType(Args);
1408e5dd7070Spatrick 
1409e5dd7070Spatrick   // Darwin doesn't support real static executables, don't link any runtime
1410e5dd7070Spatrick   // libraries with -static.
1411e5dd7070Spatrick   if (Args.hasArg(options::OPT_static) ||
1412e5dd7070Spatrick       Args.hasArg(options::OPT_fapple_kext) ||
1413e5dd7070Spatrick       Args.hasArg(options::OPT_mkernel)) {
1414e5dd7070Spatrick     if (ForceLinkBuiltinRT)
1415e5dd7070Spatrick       AddLinkRuntimeLib(Args, CmdArgs, "builtins");
1416e5dd7070Spatrick     return;
1417e5dd7070Spatrick   }
1418e5dd7070Spatrick 
1419e5dd7070Spatrick   // Reject -static-libgcc for now, we can deal with this when and if someone
1420e5dd7070Spatrick   // cares. This is useful in situations where someone wants to statically link
1421e5dd7070Spatrick   // something like libstdc++, and needs its runtime support routines.
1422e5dd7070Spatrick   if (const Arg *A = Args.getLastArg(options::OPT_static_libgcc)) {
1423e5dd7070Spatrick     getDriver().Diag(diag::err_drv_unsupported_opt) << A->getAsString(Args);
1424e5dd7070Spatrick     return;
1425e5dd7070Spatrick   }
1426e5dd7070Spatrick 
1427*12c85518Srobert   const SanitizerArgs &Sanitize = getSanitizerArgs(Args);
1428*12c85518Srobert 
1429*12c85518Srobert   if (!Sanitize.needsSharedRt() && Sanitize.needsUbsanRt()) {
1430*12c85518Srobert     getDriver().Diag(diag::err_drv_unsupported_static_ubsan_darwin);
1431*12c85518Srobert     return;
1432*12c85518Srobert   }
1433*12c85518Srobert 
1434e5dd7070Spatrick   if (Sanitize.needsAsanRt())
1435e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs, "asan");
1436e5dd7070Spatrick   if (Sanitize.needsLsanRt())
1437e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs, "lsan");
1438*12c85518Srobert   if (Sanitize.needsUbsanRt()) {
1439*12c85518Srobert     assert(Sanitize.needsSharedRt() && "Static sanitizer runtimes not supported");
1440e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs,
1441e5dd7070Spatrick                             Sanitize.requiresMinimalRuntime() ? "ubsan_minimal"
1442*12c85518Srobert                                                               : "ubsan");
1443*12c85518Srobert   }
1444e5dd7070Spatrick   if (Sanitize.needsTsanRt())
1445e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs, "tsan");
1446e5dd7070Spatrick   if (Sanitize.needsFuzzer() && !Args.hasArg(options::OPT_dynamiclib)) {
1447e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs, "fuzzer", /*shared=*/false);
1448e5dd7070Spatrick 
1449e5dd7070Spatrick     // Libfuzzer is written in C++ and requires libcxx.
1450e5dd7070Spatrick     AddCXXStdlibLibArgs(Args, CmdArgs);
1451e5dd7070Spatrick   }
1452e5dd7070Spatrick   if (Sanitize.needsStatsRt()) {
1453e5dd7070Spatrick     AddLinkRuntimeLib(Args, CmdArgs, "stats_client", RLO_AlwaysLink);
1454e5dd7070Spatrick     AddLinkSanitizerLibArgs(Args, CmdArgs, "stats");
1455e5dd7070Spatrick   }
1456e5dd7070Spatrick 
1457e5dd7070Spatrick   const XRayArgs &XRay = getXRayArgs();
1458e5dd7070Spatrick   if (XRay.needsXRayRt()) {
1459e5dd7070Spatrick     AddLinkRuntimeLib(Args, CmdArgs, "xray");
1460e5dd7070Spatrick     AddLinkRuntimeLib(Args, CmdArgs, "xray-basic");
1461e5dd7070Spatrick     AddLinkRuntimeLib(Args, CmdArgs, "xray-fdr");
1462e5dd7070Spatrick   }
1463e5dd7070Spatrick 
1464*12c85518Srobert   if (isTargetDriverKit() && !Args.hasArg(options::OPT_nodriverkitlib)) {
1465*12c85518Srobert     CmdArgs.push_back("-framework");
1466*12c85518Srobert     CmdArgs.push_back("DriverKit");
1467*12c85518Srobert   }
1468*12c85518Srobert 
1469e5dd7070Spatrick   // Otherwise link libSystem, then the dynamic runtime library, and finally any
1470e5dd7070Spatrick   // target specific static runtime library.
1471*12c85518Srobert   if (!isTargetDriverKit())
1472e5dd7070Spatrick     CmdArgs.push_back("-lSystem");
1473e5dd7070Spatrick 
1474e5dd7070Spatrick   // Select the dynamic runtime library and the target specific static library.
1475e5dd7070Spatrick   if (isTargetIOSBased()) {
1476e5dd7070Spatrick     // If we are compiling as iOS / simulator, don't attempt to link libgcc_s.1,
1477e5dd7070Spatrick     // it never went into the SDK.
1478e5dd7070Spatrick     // Linking against libgcc_s.1 isn't needed for iOS 5.0+
1479e5dd7070Spatrick     if (isIPhoneOSVersionLT(5, 0) && !isTargetIOSSimulator() &&
1480e5dd7070Spatrick         getTriple().getArch() != llvm::Triple::aarch64)
1481e5dd7070Spatrick       CmdArgs.push_back("-lgcc_s.1");
1482e5dd7070Spatrick   }
1483e5dd7070Spatrick   AddLinkRuntimeLib(Args, CmdArgs, "builtins");
1484e5dd7070Spatrick }
1485e5dd7070Spatrick 
1486e5dd7070Spatrick /// Returns the most appropriate macOS target version for the current process.
1487e5dd7070Spatrick ///
1488e5dd7070Spatrick /// If the macOS SDK version is the same or earlier than the system version,
1489e5dd7070Spatrick /// then the SDK version is returned. Otherwise the system version is returned.
getSystemOrSDKMacOSVersion(StringRef MacOSSDKVersion)1490e5dd7070Spatrick static std::string getSystemOrSDKMacOSVersion(StringRef MacOSSDKVersion) {
1491e5dd7070Spatrick   llvm::Triple SystemTriple(llvm::sys::getProcessTriple());
1492e5dd7070Spatrick   if (!SystemTriple.isMacOSX())
1493ec727ea7Spatrick     return std::string(MacOSSDKVersion);
1494*12c85518Srobert   VersionTuple SystemVersion;
1495*12c85518Srobert   SystemTriple.getMacOSXVersion(SystemVersion);
1496*12c85518Srobert 
1497*12c85518Srobert   unsigned Major, Minor, Micro;
1498e5dd7070Spatrick   bool HadExtra;
1499e5dd7070Spatrick   if (!Driver::GetReleaseVersion(MacOSSDKVersion, Major, Minor, Micro,
1500e5dd7070Spatrick                                  HadExtra))
1501ec727ea7Spatrick     return std::string(MacOSSDKVersion);
1502e5dd7070Spatrick   VersionTuple SDKVersion(Major, Minor, Micro);
1503*12c85518Srobert 
1504e5dd7070Spatrick   if (SDKVersion > SystemVersion)
1505e5dd7070Spatrick     return SystemVersion.getAsString();
1506ec727ea7Spatrick   return std::string(MacOSSDKVersion);
1507e5dd7070Spatrick }
1508e5dd7070Spatrick 
1509e5dd7070Spatrick namespace {
1510e5dd7070Spatrick 
1511e5dd7070Spatrick /// The Darwin OS that was selected or inferred from arguments / environment.
1512e5dd7070Spatrick struct DarwinPlatform {
1513e5dd7070Spatrick   enum SourceKind {
1514e5dd7070Spatrick     /// The OS was specified using the -target argument.
1515e5dd7070Spatrick     TargetArg,
1516*12c85518Srobert     /// The OS was specified using the -mtargetos= argument.
1517*12c85518Srobert     MTargetOSArg,
1518e5dd7070Spatrick     /// The OS was specified using the -m<os>-version-min argument.
1519e5dd7070Spatrick     OSVersionArg,
1520e5dd7070Spatrick     /// The OS was specified using the OS_DEPLOYMENT_TARGET environment.
1521e5dd7070Spatrick     DeploymentTargetEnv,
1522e5dd7070Spatrick     /// The OS was inferred from the SDK.
1523e5dd7070Spatrick     InferredFromSDK,
1524e5dd7070Spatrick     /// The OS was inferred from the -arch.
1525e5dd7070Spatrick     InferredFromArch
1526e5dd7070Spatrick   };
1527e5dd7070Spatrick 
1528e5dd7070Spatrick   using DarwinPlatformKind = Darwin::DarwinPlatformKind;
1529e5dd7070Spatrick   using DarwinEnvironmentKind = Darwin::DarwinEnvironmentKind;
1530e5dd7070Spatrick 
getPlatform__anon619bc4d40311::DarwinPlatform1531e5dd7070Spatrick   DarwinPlatformKind getPlatform() const { return Platform; }
1532e5dd7070Spatrick 
getEnvironment__anon619bc4d40311::DarwinPlatform1533e5dd7070Spatrick   DarwinEnvironmentKind getEnvironment() const { return Environment; }
1534e5dd7070Spatrick 
setEnvironment__anon619bc4d40311::DarwinPlatform1535e5dd7070Spatrick   void setEnvironment(DarwinEnvironmentKind Kind) {
1536e5dd7070Spatrick     Environment = Kind;
1537e5dd7070Spatrick     InferSimulatorFromArch = false;
1538e5dd7070Spatrick   }
1539e5dd7070Spatrick 
getOSVersion__anon619bc4d40311::DarwinPlatform1540e5dd7070Spatrick   StringRef getOSVersion() const {
1541e5dd7070Spatrick     if (Kind == OSVersionArg)
1542e5dd7070Spatrick       return Argument->getValue();
1543e5dd7070Spatrick     return OSVersion;
1544e5dd7070Spatrick   }
1545e5dd7070Spatrick 
setOSVersion__anon619bc4d40311::DarwinPlatform1546e5dd7070Spatrick   void setOSVersion(StringRef S) {
1547e5dd7070Spatrick     assert(Kind == TargetArg && "Unexpected kind!");
1548ec727ea7Spatrick     OSVersion = std::string(S);
1549e5dd7070Spatrick   }
1550e5dd7070Spatrick 
hasOSVersion__anon619bc4d40311::DarwinPlatform1551e5dd7070Spatrick   bool hasOSVersion() const { return HasOSVersion; }
1552e5dd7070Spatrick 
getNativeTargetVersion__anon619bc4d40311::DarwinPlatform1553a9ac8606Spatrick   VersionTuple getNativeTargetVersion() const {
1554a9ac8606Spatrick     assert(Environment == DarwinEnvironmentKind::MacCatalyst &&
1555a9ac8606Spatrick            "native target version is specified only for Mac Catalyst");
1556a9ac8606Spatrick     return NativeTargetVersion;
1557a9ac8606Spatrick   }
1558a9ac8606Spatrick 
1559e5dd7070Spatrick   /// Returns true if the target OS was explicitly specified.
isExplicitlySpecified__anon619bc4d40311::DarwinPlatform1560e5dd7070Spatrick   bool isExplicitlySpecified() const { return Kind <= DeploymentTargetEnv; }
1561e5dd7070Spatrick 
1562e5dd7070Spatrick   /// Returns true if the simulator environment can be inferred from the arch.
canInferSimulatorFromArch__anon619bc4d40311::DarwinPlatform1563e5dd7070Spatrick   bool canInferSimulatorFromArch() const { return InferSimulatorFromArch; }
1564e5dd7070Spatrick 
getTargetVariantTriple__anon619bc4d40311::DarwinPlatform1565*12c85518Srobert   const std::optional<llvm::Triple> &getTargetVariantTriple() const {
1566*12c85518Srobert     return TargetVariantTriple;
1567*12c85518Srobert   }
1568*12c85518Srobert 
1569e5dd7070Spatrick   /// Adds the -m<os>-version-min argument to the compiler invocation.
addOSVersionMinArgument__anon619bc4d40311::DarwinPlatform1570e5dd7070Spatrick   void addOSVersionMinArgument(DerivedArgList &Args, const OptTable &Opts) {
1571e5dd7070Spatrick     if (Argument)
1572e5dd7070Spatrick       return;
1573*12c85518Srobert     assert(Kind != TargetArg && Kind != MTargetOSArg && Kind != OSVersionArg &&
1574*12c85518Srobert            "Invalid kind");
1575e5dd7070Spatrick     options::ID Opt;
1576e5dd7070Spatrick     switch (Platform) {
1577e5dd7070Spatrick     case DarwinPlatformKind::MacOS:
1578*12c85518Srobert       Opt = options::OPT_mmacos_version_min_EQ;
1579e5dd7070Spatrick       break;
1580e5dd7070Spatrick     case DarwinPlatformKind::IPhoneOS:
1581*12c85518Srobert       Opt = options::OPT_mios_version_min_EQ;
1582e5dd7070Spatrick       break;
1583e5dd7070Spatrick     case DarwinPlatformKind::TvOS:
1584e5dd7070Spatrick       Opt = options::OPT_mtvos_version_min_EQ;
1585e5dd7070Spatrick       break;
1586e5dd7070Spatrick     case DarwinPlatformKind::WatchOS:
1587e5dd7070Spatrick       Opt = options::OPT_mwatchos_version_min_EQ;
1588e5dd7070Spatrick       break;
1589*12c85518Srobert     case DarwinPlatformKind::DriverKit:
1590*12c85518Srobert       // DriverKit always explicitly provides a version in the triple.
1591*12c85518Srobert       return;
1592e5dd7070Spatrick     }
1593e5dd7070Spatrick     Argument = Args.MakeJoinedArg(nullptr, Opts.getOption(Opt), OSVersion);
1594e5dd7070Spatrick     Args.append(Argument);
1595e5dd7070Spatrick   }
1596e5dd7070Spatrick 
1597e5dd7070Spatrick   /// Returns the OS version with the argument / environment variable that
1598e5dd7070Spatrick   /// specified it.
getAsString__anon619bc4d40311::DarwinPlatform1599e5dd7070Spatrick   std::string getAsString(DerivedArgList &Args, const OptTable &Opts) {
1600e5dd7070Spatrick     switch (Kind) {
1601e5dd7070Spatrick     case TargetArg:
1602*12c85518Srobert     case MTargetOSArg:
1603e5dd7070Spatrick     case OSVersionArg:
1604e5dd7070Spatrick     case InferredFromSDK:
1605e5dd7070Spatrick     case InferredFromArch:
1606e5dd7070Spatrick       assert(Argument && "OS version argument not yet inferred");
1607e5dd7070Spatrick       return Argument->getAsString(Args);
1608e5dd7070Spatrick     case DeploymentTargetEnv:
1609e5dd7070Spatrick       return (llvm::Twine(EnvVarName) + "=" + OSVersion).str();
1610e5dd7070Spatrick     }
1611e5dd7070Spatrick     llvm_unreachable("Unsupported Darwin Source Kind");
1612e5dd7070Spatrick   }
1613e5dd7070Spatrick 
setEnvironment__anon619bc4d40311::DarwinPlatform1614*12c85518Srobert   void setEnvironment(llvm::Triple::EnvironmentType EnvType,
1615*12c85518Srobert                       const VersionTuple &OSVersion,
1616*12c85518Srobert                       const std::optional<DarwinSDKInfo> &SDKInfo) {
1617*12c85518Srobert     switch (EnvType) {
1618a9ac8606Spatrick     case llvm::Triple::Simulator:
1619*12c85518Srobert       Environment = DarwinEnvironmentKind::Simulator;
1620a9ac8606Spatrick       break;
1621a9ac8606Spatrick     case llvm::Triple::MacABI: {
1622*12c85518Srobert       Environment = DarwinEnvironmentKind::MacCatalyst;
1623a9ac8606Spatrick       // The minimum native macOS target for MacCatalyst is macOS 10.15.
1624*12c85518Srobert       NativeTargetVersion = VersionTuple(10, 15);
1625*12c85518Srobert       if (HasOSVersion && SDKInfo) {
1626a9ac8606Spatrick         if (const auto *MacCatalystToMacOSMapping = SDKInfo->getVersionMapping(
1627a9ac8606Spatrick                 DarwinSDKInfo::OSEnvPair::macCatalystToMacOSPair())) {
1628a9ac8606Spatrick           if (auto MacOSVersion = MacCatalystToMacOSMapping->map(
1629*12c85518Srobert                   OSVersion, NativeTargetVersion, std::nullopt)) {
1630a9ac8606Spatrick             NativeTargetVersion = *MacOSVersion;
1631a9ac8606Spatrick           }
1632a9ac8606Spatrick         }
1633a9ac8606Spatrick       }
1634*12c85518Srobert       // In a zippered build, we could be building for a macOS target that's
1635*12c85518Srobert       // lower than the version that's implied by the OS version. In that case
1636*12c85518Srobert       // we need to use the minimum version as the native target version.
1637*12c85518Srobert       if (TargetVariantTriple) {
1638*12c85518Srobert         auto TargetVariantVersion = TargetVariantTriple->getOSVersion();
1639*12c85518Srobert         if (TargetVariantVersion.getMajor()) {
1640*12c85518Srobert           if (TargetVariantVersion < NativeTargetVersion)
1641*12c85518Srobert             NativeTargetVersion = TargetVariantVersion;
1642*12c85518Srobert         }
1643*12c85518Srobert       }
1644a9ac8606Spatrick       break;
1645a9ac8606Spatrick     }
1646a9ac8606Spatrick     default:
1647a9ac8606Spatrick       break;
1648a9ac8606Spatrick     }
1649*12c85518Srobert   }
1650*12c85518Srobert 
1651*12c85518Srobert   static DarwinPlatform
createFromTarget__anon619bc4d40311::DarwinPlatform1652*12c85518Srobert   createFromTarget(const llvm::Triple &TT, StringRef OSVersion, Arg *A,
1653*12c85518Srobert                    std::optional<llvm::Triple> TargetVariantTriple,
1654*12c85518Srobert                    const std::optional<DarwinSDKInfo> &SDKInfo) {
1655*12c85518Srobert     DarwinPlatform Result(TargetArg, getPlatformFromOS(TT.getOS()), OSVersion,
1656*12c85518Srobert                           A);
1657*12c85518Srobert     VersionTuple OsVersion = TT.getOSVersion();
1658*12c85518Srobert     if (OsVersion.getMajor() == 0)
1659*12c85518Srobert       Result.HasOSVersion = false;
1660*12c85518Srobert     Result.TargetVariantTriple = TargetVariantTriple;
1661*12c85518Srobert     Result.setEnvironment(TT.getEnvironment(), OsVersion, SDKInfo);
1662e5dd7070Spatrick     return Result;
1663e5dd7070Spatrick   }
1664*12c85518Srobert   static DarwinPlatform
createFromMTargetOS__anon619bc4d40311::DarwinPlatform1665*12c85518Srobert   createFromMTargetOS(llvm::Triple::OSType OS, VersionTuple OSVersion,
1666*12c85518Srobert                       llvm::Triple::EnvironmentType Environment, Arg *A,
1667*12c85518Srobert                       const std::optional<DarwinSDKInfo> &SDKInfo) {
1668*12c85518Srobert     DarwinPlatform Result(MTargetOSArg, getPlatformFromOS(OS),
1669*12c85518Srobert                           OSVersion.getAsString(), A);
1670*12c85518Srobert     Result.InferSimulatorFromArch = false;
1671*12c85518Srobert     Result.setEnvironment(Environment, OSVersion, SDKInfo);
1672*12c85518Srobert     return Result;
1673*12c85518Srobert   }
createOSVersionArg__anon619bc4d40311::DarwinPlatform1674*12c85518Srobert   static DarwinPlatform createOSVersionArg(DarwinPlatformKind Platform, Arg *A,
1675*12c85518Srobert                                            bool IsSimulator) {
1676*12c85518Srobert     DarwinPlatform Result{OSVersionArg, Platform, A};
1677*12c85518Srobert     if (IsSimulator)
1678*12c85518Srobert       Result.Environment = DarwinEnvironmentKind::Simulator;
1679*12c85518Srobert     return Result;
1680e5dd7070Spatrick   }
createDeploymentTargetEnv__anon619bc4d40311::DarwinPlatform1681e5dd7070Spatrick   static DarwinPlatform createDeploymentTargetEnv(DarwinPlatformKind Platform,
1682e5dd7070Spatrick                                                   StringRef EnvVarName,
1683e5dd7070Spatrick                                                   StringRef Value) {
1684e5dd7070Spatrick     DarwinPlatform Result(DeploymentTargetEnv, Platform, Value);
1685e5dd7070Spatrick     Result.EnvVarName = EnvVarName;
1686e5dd7070Spatrick     return Result;
1687e5dd7070Spatrick   }
createFromSDK__anon619bc4d40311::DarwinPlatform1688e5dd7070Spatrick   static DarwinPlatform createFromSDK(DarwinPlatformKind Platform,
1689e5dd7070Spatrick                                       StringRef Value,
1690e5dd7070Spatrick                                       bool IsSimulator = false) {
1691e5dd7070Spatrick     DarwinPlatform Result(InferredFromSDK, Platform, Value);
1692e5dd7070Spatrick     if (IsSimulator)
1693e5dd7070Spatrick       Result.Environment = DarwinEnvironmentKind::Simulator;
1694e5dd7070Spatrick     Result.InferSimulatorFromArch = false;
1695e5dd7070Spatrick     return Result;
1696e5dd7070Spatrick   }
createFromArch__anon619bc4d40311::DarwinPlatform1697e5dd7070Spatrick   static DarwinPlatform createFromArch(llvm::Triple::OSType OS,
1698e5dd7070Spatrick                                        StringRef Value) {
1699e5dd7070Spatrick     return DarwinPlatform(InferredFromArch, getPlatformFromOS(OS), Value);
1700e5dd7070Spatrick   }
1701e5dd7070Spatrick 
1702e5dd7070Spatrick   /// Constructs an inferred SDKInfo value based on the version inferred from
1703e5dd7070Spatrick   /// the SDK path itself. Only works for values that were created by inferring
1704e5dd7070Spatrick   /// the platform from the SDKPath.
inferSDKInfo__anon619bc4d40311::DarwinPlatform1705e5dd7070Spatrick   DarwinSDKInfo inferSDKInfo() {
1706e5dd7070Spatrick     assert(Kind == InferredFromSDK && "can infer SDK info only");
1707e5dd7070Spatrick     llvm::VersionTuple Version;
1708e5dd7070Spatrick     bool IsValid = !Version.tryParse(OSVersion);
1709e5dd7070Spatrick     (void)IsValid;
1710e5dd7070Spatrick     assert(IsValid && "invalid SDK version");
1711a9ac8606Spatrick     return DarwinSDKInfo(
1712a9ac8606Spatrick         Version,
1713a9ac8606Spatrick         /*MaximumDeploymentTarget=*/VersionTuple(Version.getMajor(), 0, 99));
1714e5dd7070Spatrick   }
1715e5dd7070Spatrick 
1716e5dd7070Spatrick private:
DarwinPlatform__anon619bc4d40311::DarwinPlatform1717e5dd7070Spatrick   DarwinPlatform(SourceKind Kind, DarwinPlatformKind Platform, Arg *Argument)
1718e5dd7070Spatrick       : Kind(Kind), Platform(Platform), Argument(Argument) {}
DarwinPlatform__anon619bc4d40311::DarwinPlatform1719e5dd7070Spatrick   DarwinPlatform(SourceKind Kind, DarwinPlatformKind Platform, StringRef Value,
1720e5dd7070Spatrick                  Arg *Argument = nullptr)
1721e5dd7070Spatrick       : Kind(Kind), Platform(Platform), OSVersion(Value), Argument(Argument) {}
1722e5dd7070Spatrick 
getPlatformFromOS__anon619bc4d40311::DarwinPlatform1723e5dd7070Spatrick   static DarwinPlatformKind getPlatformFromOS(llvm::Triple::OSType OS) {
1724e5dd7070Spatrick     switch (OS) {
1725e5dd7070Spatrick     case llvm::Triple::Darwin:
1726e5dd7070Spatrick     case llvm::Triple::MacOSX:
1727e5dd7070Spatrick       return DarwinPlatformKind::MacOS;
1728e5dd7070Spatrick     case llvm::Triple::IOS:
1729e5dd7070Spatrick       return DarwinPlatformKind::IPhoneOS;
1730e5dd7070Spatrick     case llvm::Triple::TvOS:
1731e5dd7070Spatrick       return DarwinPlatformKind::TvOS;
1732e5dd7070Spatrick     case llvm::Triple::WatchOS:
1733e5dd7070Spatrick       return DarwinPlatformKind::WatchOS;
1734*12c85518Srobert     case llvm::Triple::DriverKit:
1735*12c85518Srobert       return DarwinPlatformKind::DriverKit;
1736e5dd7070Spatrick     default:
1737e5dd7070Spatrick       llvm_unreachable("Unable to infer Darwin variant");
1738e5dd7070Spatrick     }
1739e5dd7070Spatrick   }
1740e5dd7070Spatrick 
1741e5dd7070Spatrick   SourceKind Kind;
1742e5dd7070Spatrick   DarwinPlatformKind Platform;
1743e5dd7070Spatrick   DarwinEnvironmentKind Environment = DarwinEnvironmentKind::NativeEnvironment;
1744a9ac8606Spatrick   VersionTuple NativeTargetVersion;
1745e5dd7070Spatrick   std::string OSVersion;
1746e5dd7070Spatrick   bool HasOSVersion = true, InferSimulatorFromArch = true;
1747e5dd7070Spatrick   Arg *Argument;
1748e5dd7070Spatrick   StringRef EnvVarName;
1749*12c85518Srobert   std::optional<llvm::Triple> TargetVariantTriple;
1750e5dd7070Spatrick };
1751e5dd7070Spatrick 
1752e5dd7070Spatrick /// Returns the deployment target that's specified using the -m<os>-version-min
1753e5dd7070Spatrick /// argument.
1754*12c85518Srobert std::optional<DarwinPlatform>
getDeploymentTargetFromOSVersionArg(DerivedArgList & Args,const Driver & TheDriver)1755e5dd7070Spatrick getDeploymentTargetFromOSVersionArg(DerivedArgList &Args,
1756e5dd7070Spatrick                                     const Driver &TheDriver) {
1757*12c85518Srobert   Arg *macOSVersion = Args.getLastArg(options::OPT_mmacos_version_min_EQ);
1758*12c85518Srobert   Arg *iOSVersion = Args.getLastArg(options::OPT_mios_version_min_EQ,
1759e5dd7070Spatrick                                     options::OPT_mios_simulator_version_min_EQ);
1760e5dd7070Spatrick   Arg *TvOSVersion =
1761e5dd7070Spatrick       Args.getLastArg(options::OPT_mtvos_version_min_EQ,
1762e5dd7070Spatrick                       options::OPT_mtvos_simulator_version_min_EQ);
1763e5dd7070Spatrick   Arg *WatchOSVersion =
1764e5dd7070Spatrick       Args.getLastArg(options::OPT_mwatchos_version_min_EQ,
1765e5dd7070Spatrick                       options::OPT_mwatchos_simulator_version_min_EQ);
1766*12c85518Srobert   if (macOSVersion) {
1767e5dd7070Spatrick     if (iOSVersion || TvOSVersion || WatchOSVersion) {
1768e5dd7070Spatrick       TheDriver.Diag(diag::err_drv_argument_not_allowed_with)
1769*12c85518Srobert           << macOSVersion->getAsString(Args)
1770e5dd7070Spatrick           << (iOSVersion ? iOSVersion
1771e5dd7070Spatrick                          : TvOSVersion ? TvOSVersion : WatchOSVersion)
1772e5dd7070Spatrick                  ->getAsString(Args);
1773e5dd7070Spatrick     }
1774*12c85518Srobert     return DarwinPlatform::createOSVersionArg(Darwin::MacOS, macOSVersion,
1775*12c85518Srobert                                               /*IsImulator=*/false);
1776e5dd7070Spatrick   } else if (iOSVersion) {
1777e5dd7070Spatrick     if (TvOSVersion || WatchOSVersion) {
1778e5dd7070Spatrick       TheDriver.Diag(diag::err_drv_argument_not_allowed_with)
1779e5dd7070Spatrick           << iOSVersion->getAsString(Args)
1780e5dd7070Spatrick           << (TvOSVersion ? TvOSVersion : WatchOSVersion)->getAsString(Args);
1781e5dd7070Spatrick     }
1782*12c85518Srobert     return DarwinPlatform::createOSVersionArg(
1783*12c85518Srobert         Darwin::IPhoneOS, iOSVersion,
1784*12c85518Srobert         iOSVersion->getOption().getID() ==
1785*12c85518Srobert             options::OPT_mios_simulator_version_min_EQ);
1786e5dd7070Spatrick   } else if (TvOSVersion) {
1787e5dd7070Spatrick     if (WatchOSVersion) {
1788e5dd7070Spatrick       TheDriver.Diag(diag::err_drv_argument_not_allowed_with)
1789e5dd7070Spatrick           << TvOSVersion->getAsString(Args)
1790e5dd7070Spatrick           << WatchOSVersion->getAsString(Args);
1791e5dd7070Spatrick     }
1792*12c85518Srobert     return DarwinPlatform::createOSVersionArg(
1793*12c85518Srobert         Darwin::TvOS, TvOSVersion,
1794*12c85518Srobert         TvOSVersion->getOption().getID() ==
1795*12c85518Srobert             options::OPT_mtvos_simulator_version_min_EQ);
1796e5dd7070Spatrick   } else if (WatchOSVersion)
1797*12c85518Srobert     return DarwinPlatform::createOSVersionArg(
1798*12c85518Srobert         Darwin::WatchOS, WatchOSVersion,
1799*12c85518Srobert         WatchOSVersion->getOption().getID() ==
1800*12c85518Srobert             options::OPT_mwatchos_simulator_version_min_EQ);
1801*12c85518Srobert   return std::nullopt;
1802e5dd7070Spatrick }
1803e5dd7070Spatrick 
1804e5dd7070Spatrick /// Returns the deployment target that's specified using the
1805e5dd7070Spatrick /// OS_DEPLOYMENT_TARGET environment variable.
1806*12c85518Srobert std::optional<DarwinPlatform>
getDeploymentTargetFromEnvironmentVariables(const Driver & TheDriver,const llvm::Triple & Triple)1807e5dd7070Spatrick getDeploymentTargetFromEnvironmentVariables(const Driver &TheDriver,
1808e5dd7070Spatrick                                             const llvm::Triple &Triple) {
1809e5dd7070Spatrick   std::string Targets[Darwin::LastDarwinPlatform + 1];
1810e5dd7070Spatrick   const char *EnvVars[] = {
1811e5dd7070Spatrick       "MACOSX_DEPLOYMENT_TARGET",
1812e5dd7070Spatrick       "IPHONEOS_DEPLOYMENT_TARGET",
1813e5dd7070Spatrick       "TVOS_DEPLOYMENT_TARGET",
1814e5dd7070Spatrick       "WATCHOS_DEPLOYMENT_TARGET",
1815*12c85518Srobert       "DRIVERKIT_DEPLOYMENT_TARGET",
1816e5dd7070Spatrick   };
1817*12c85518Srobert   static_assert(std::size(EnvVars) == Darwin::LastDarwinPlatform + 1,
1818e5dd7070Spatrick                 "Missing platform");
1819*12c85518Srobert   for (const auto &I : llvm::enumerate(llvm::ArrayRef(EnvVars))) {
1820e5dd7070Spatrick     if (char *Env = ::getenv(I.value()))
1821e5dd7070Spatrick       Targets[I.index()] = Env;
1822e5dd7070Spatrick   }
1823e5dd7070Spatrick 
1824e5dd7070Spatrick   // Allow conflicts among OSX and iOS for historical reasons, but choose the
1825e5dd7070Spatrick   // default platform.
1826e5dd7070Spatrick   if (!Targets[Darwin::MacOS].empty() &&
1827e5dd7070Spatrick       (!Targets[Darwin::IPhoneOS].empty() ||
1828e5dd7070Spatrick        !Targets[Darwin::WatchOS].empty() || !Targets[Darwin::TvOS].empty())) {
1829e5dd7070Spatrick     if (Triple.getArch() == llvm::Triple::arm ||
1830e5dd7070Spatrick         Triple.getArch() == llvm::Triple::aarch64 ||
1831e5dd7070Spatrick         Triple.getArch() == llvm::Triple::thumb)
1832e5dd7070Spatrick       Targets[Darwin::MacOS] = "";
1833e5dd7070Spatrick     else
1834e5dd7070Spatrick       Targets[Darwin::IPhoneOS] = Targets[Darwin::WatchOS] =
1835e5dd7070Spatrick           Targets[Darwin::TvOS] = "";
1836e5dd7070Spatrick   } else {
1837e5dd7070Spatrick     // Don't allow conflicts in any other platform.
1838*12c85518Srobert     unsigned FirstTarget = std::size(Targets);
1839*12c85518Srobert     for (unsigned I = 0; I != std::size(Targets); ++I) {
1840e5dd7070Spatrick       if (Targets[I].empty())
1841e5dd7070Spatrick         continue;
1842*12c85518Srobert       if (FirstTarget == std::size(Targets))
1843e5dd7070Spatrick         FirstTarget = I;
1844e5dd7070Spatrick       else
1845e5dd7070Spatrick         TheDriver.Diag(diag::err_drv_conflicting_deployment_targets)
1846e5dd7070Spatrick             << Targets[FirstTarget] << Targets[I];
1847e5dd7070Spatrick     }
1848e5dd7070Spatrick   }
1849e5dd7070Spatrick 
1850*12c85518Srobert   for (const auto &Target : llvm::enumerate(llvm::ArrayRef(Targets))) {
1851e5dd7070Spatrick     if (!Target.value().empty())
1852e5dd7070Spatrick       return DarwinPlatform::createDeploymentTargetEnv(
1853e5dd7070Spatrick           (Darwin::DarwinPlatformKind)Target.index(), EnvVars[Target.index()],
1854e5dd7070Spatrick           Target.value());
1855e5dd7070Spatrick   }
1856*12c85518Srobert   return std::nullopt;
1857e5dd7070Spatrick }
1858e5dd7070Spatrick 
1859a9ac8606Spatrick /// Returns the SDK name without the optional prefix that ends with a '.' or an
1860a9ac8606Spatrick /// empty string otherwise.
dropSDKNamePrefix(StringRef SDKName)1861a9ac8606Spatrick static StringRef dropSDKNamePrefix(StringRef SDKName) {
1862a9ac8606Spatrick   size_t PrefixPos = SDKName.find('.');
1863a9ac8606Spatrick   if (PrefixPos == StringRef::npos)
1864a9ac8606Spatrick     return "";
1865a9ac8606Spatrick   return SDKName.substr(PrefixPos + 1);
1866a9ac8606Spatrick }
1867a9ac8606Spatrick 
1868e5dd7070Spatrick /// Tries to infer the deployment target from the SDK specified by -isysroot
1869e5dd7070Spatrick /// (or SDKROOT). Uses the version specified in the SDKSettings.json file if
1870e5dd7070Spatrick /// it's available.
1871*12c85518Srobert std::optional<DarwinPlatform>
inferDeploymentTargetFromSDK(DerivedArgList & Args,const std::optional<DarwinSDKInfo> & SDKInfo)1872e5dd7070Spatrick inferDeploymentTargetFromSDK(DerivedArgList &Args,
1873*12c85518Srobert                              const std::optional<DarwinSDKInfo> &SDKInfo) {
1874e5dd7070Spatrick   const Arg *A = Args.getLastArg(options::OPT_isysroot);
1875e5dd7070Spatrick   if (!A)
1876*12c85518Srobert     return std::nullopt;
1877e5dd7070Spatrick   StringRef isysroot = A->getValue();
1878e5dd7070Spatrick   StringRef SDK = Darwin::getSDKName(isysroot);
1879e5dd7070Spatrick   if (!SDK.size())
1880*12c85518Srobert     return std::nullopt;
1881e5dd7070Spatrick 
1882e5dd7070Spatrick   std::string Version;
1883e5dd7070Spatrick   if (SDKInfo) {
1884e5dd7070Spatrick     // Get the version from the SDKSettings.json if it's available.
1885e5dd7070Spatrick     Version = SDKInfo->getVersion().getAsString();
1886e5dd7070Spatrick   } else {
1887e5dd7070Spatrick     // Slice the version number out.
1888e5dd7070Spatrick     // Version number is between the first and the last number.
1889e5dd7070Spatrick     size_t StartVer = SDK.find_first_of("0123456789");
1890e5dd7070Spatrick     size_t EndVer = SDK.find_last_of("0123456789");
1891e5dd7070Spatrick     if (StartVer != StringRef::npos && EndVer > StartVer)
1892ec727ea7Spatrick       Version = std::string(SDK.slice(StartVer, EndVer + 1));
1893e5dd7070Spatrick   }
1894e5dd7070Spatrick   if (Version.empty())
1895*12c85518Srobert     return std::nullopt;
1896e5dd7070Spatrick 
1897a9ac8606Spatrick   auto CreatePlatformFromSDKName =
1898*12c85518Srobert       [&](StringRef SDK) -> std::optional<DarwinPlatform> {
1899e5dd7070Spatrick     if (SDK.startswith("iPhoneOS") || SDK.startswith("iPhoneSimulator"))
1900e5dd7070Spatrick       return DarwinPlatform::createFromSDK(
1901e5dd7070Spatrick           Darwin::IPhoneOS, Version,
1902e5dd7070Spatrick           /*IsSimulator=*/SDK.startswith("iPhoneSimulator"));
1903e5dd7070Spatrick     else if (SDK.startswith("MacOSX"))
1904e5dd7070Spatrick       return DarwinPlatform::createFromSDK(Darwin::MacOS,
1905e5dd7070Spatrick                                            getSystemOrSDKMacOSVersion(Version));
1906e5dd7070Spatrick     else if (SDK.startswith("WatchOS") || SDK.startswith("WatchSimulator"))
1907e5dd7070Spatrick       return DarwinPlatform::createFromSDK(
1908e5dd7070Spatrick           Darwin::WatchOS, Version,
1909e5dd7070Spatrick           /*IsSimulator=*/SDK.startswith("WatchSimulator"));
1910e5dd7070Spatrick     else if (SDK.startswith("AppleTVOS") || SDK.startswith("AppleTVSimulator"))
1911e5dd7070Spatrick       return DarwinPlatform::createFromSDK(
1912e5dd7070Spatrick           Darwin::TvOS, Version,
1913e5dd7070Spatrick           /*IsSimulator=*/SDK.startswith("AppleTVSimulator"));
1914*12c85518Srobert     else if (SDK.startswith("DriverKit"))
1915*12c85518Srobert       return DarwinPlatform::createFromSDK(Darwin::DriverKit, Version);
1916*12c85518Srobert     return std::nullopt;
1917a9ac8606Spatrick   };
1918a9ac8606Spatrick   if (auto Result = CreatePlatformFromSDKName(SDK))
1919a9ac8606Spatrick     return Result;
1920a9ac8606Spatrick   // The SDK can be an SDK variant with a name like `<prefix>.<platform>`.
1921a9ac8606Spatrick   return CreatePlatformFromSDKName(dropSDKNamePrefix(SDK));
1922e5dd7070Spatrick }
1923e5dd7070Spatrick 
getOSVersion(llvm::Triple::OSType OS,const llvm::Triple & Triple,const Driver & TheDriver)1924e5dd7070Spatrick std::string getOSVersion(llvm::Triple::OSType OS, const llvm::Triple &Triple,
1925e5dd7070Spatrick                          const Driver &TheDriver) {
1926*12c85518Srobert   VersionTuple OsVersion;
1927e5dd7070Spatrick   llvm::Triple SystemTriple(llvm::sys::getProcessTriple());
1928e5dd7070Spatrick   switch (OS) {
1929e5dd7070Spatrick   case llvm::Triple::Darwin:
1930e5dd7070Spatrick   case llvm::Triple::MacOSX:
1931e5dd7070Spatrick     // If there is no version specified on triple, and both host and target are
1932e5dd7070Spatrick     // macos, use the host triple to infer OS version.
1933e5dd7070Spatrick     if (Triple.isMacOSX() && SystemTriple.isMacOSX() &&
1934e5dd7070Spatrick         !Triple.getOSMajorVersion())
1935*12c85518Srobert       SystemTriple.getMacOSXVersion(OsVersion);
1936*12c85518Srobert     else if (!Triple.getMacOSXVersion(OsVersion))
1937e5dd7070Spatrick       TheDriver.Diag(diag::err_drv_invalid_darwin_version)
1938e5dd7070Spatrick           << Triple.getOSName();
1939e5dd7070Spatrick     break;
1940e5dd7070Spatrick   case llvm::Triple::IOS:
1941*12c85518Srobert     if (Triple.isMacCatalystEnvironment() && !Triple.getOSMajorVersion()) {
1942*12c85518Srobert       OsVersion = VersionTuple(13, 1);
1943*12c85518Srobert     } else
1944*12c85518Srobert       OsVersion = Triple.getiOSVersion();
1945e5dd7070Spatrick     break;
1946e5dd7070Spatrick   case llvm::Triple::TvOS:
1947*12c85518Srobert     OsVersion = Triple.getOSVersion();
1948e5dd7070Spatrick     break;
1949e5dd7070Spatrick   case llvm::Triple::WatchOS:
1950*12c85518Srobert     OsVersion = Triple.getWatchOSVersion();
1951*12c85518Srobert     break;
1952*12c85518Srobert   case llvm::Triple::DriverKit:
1953*12c85518Srobert     OsVersion = Triple.getDriverKitVersion();
1954e5dd7070Spatrick     break;
1955e5dd7070Spatrick   default:
1956e5dd7070Spatrick     llvm_unreachable("Unexpected OS type");
1957e5dd7070Spatrick     break;
1958e5dd7070Spatrick   }
1959e5dd7070Spatrick 
1960e5dd7070Spatrick   std::string OSVersion;
1961*12c85518Srobert   llvm::raw_string_ostream(OSVersion)
1962*12c85518Srobert       << OsVersion.getMajor() << '.' << OsVersion.getMinor().value_or(0) << '.'
1963*12c85518Srobert       << OsVersion.getSubminor().value_or(0);
1964e5dd7070Spatrick   return OSVersion;
1965e5dd7070Spatrick }
1966e5dd7070Spatrick 
1967e5dd7070Spatrick /// Tries to infer the target OS from the -arch.
1968*12c85518Srobert std::optional<DarwinPlatform>
inferDeploymentTargetFromArch(DerivedArgList & Args,const Darwin & Toolchain,const llvm::Triple & Triple,const Driver & TheDriver)1969e5dd7070Spatrick inferDeploymentTargetFromArch(DerivedArgList &Args, const Darwin &Toolchain,
1970e5dd7070Spatrick                               const llvm::Triple &Triple,
1971e5dd7070Spatrick                               const Driver &TheDriver) {
1972e5dd7070Spatrick   llvm::Triple::OSType OSTy = llvm::Triple::UnknownOS;
1973e5dd7070Spatrick 
1974e5dd7070Spatrick   StringRef MachOArchName = Toolchain.getMachOArchName(Args);
1975*12c85518Srobert   if (MachOArchName == "arm64" || MachOArchName == "arm64e")
1976ec727ea7Spatrick     OSTy = llvm::Triple::MacOSX;
1977*12c85518Srobert   else if (MachOArchName == "armv7" || MachOArchName == "armv7s")
1978e5dd7070Spatrick     OSTy = llvm::Triple::IOS;
1979e5dd7070Spatrick   else if (MachOArchName == "armv7k" || MachOArchName == "arm64_32")
1980e5dd7070Spatrick     OSTy = llvm::Triple::WatchOS;
1981e5dd7070Spatrick   else if (MachOArchName != "armv6m" && MachOArchName != "armv7m" &&
1982e5dd7070Spatrick            MachOArchName != "armv7em")
1983e5dd7070Spatrick     OSTy = llvm::Triple::MacOSX;
1984e5dd7070Spatrick   if (OSTy == llvm::Triple::UnknownOS)
1985*12c85518Srobert     return std::nullopt;
1986e5dd7070Spatrick   return DarwinPlatform::createFromArch(OSTy,
1987e5dd7070Spatrick                                         getOSVersion(OSTy, Triple, TheDriver));
1988e5dd7070Spatrick }
1989e5dd7070Spatrick 
1990e5dd7070Spatrick /// Returns the deployment target that's specified using the -target option.
getDeploymentTargetFromTargetArg(DerivedArgList & Args,const llvm::Triple & Triple,const Driver & TheDriver,const std::optional<DarwinSDKInfo> & SDKInfo)1991*12c85518Srobert std::optional<DarwinPlatform> getDeploymentTargetFromTargetArg(
1992a9ac8606Spatrick     DerivedArgList &Args, const llvm::Triple &Triple, const Driver &TheDriver,
1993*12c85518Srobert     const std::optional<DarwinSDKInfo> &SDKInfo) {
1994e5dd7070Spatrick   if (!Args.hasArg(options::OPT_target))
1995*12c85518Srobert     return std::nullopt;
1996e5dd7070Spatrick   if (Triple.getOS() == llvm::Triple::Darwin ||
1997e5dd7070Spatrick       Triple.getOS() == llvm::Triple::UnknownOS)
1998*12c85518Srobert     return std::nullopt;
1999e5dd7070Spatrick   std::string OSVersion = getOSVersion(Triple.getOS(), Triple, TheDriver);
2000*12c85518Srobert   std::optional<llvm::Triple> TargetVariantTriple;
2001*12c85518Srobert   for (const Arg *A : Args.filtered(options::OPT_darwin_target_variant)) {
2002*12c85518Srobert     llvm::Triple TVT(A->getValue());
2003*12c85518Srobert     // Find a matching <arch>-<vendor> target variant triple that can be used.
2004*12c85518Srobert     if ((Triple.getArch() == llvm::Triple::aarch64 ||
2005*12c85518Srobert          TVT.getArchName() == Triple.getArchName()) &&
2006*12c85518Srobert         TVT.getArch() == Triple.getArch() &&
2007*12c85518Srobert         TVT.getSubArch() == Triple.getSubArch() &&
2008*12c85518Srobert         TVT.getVendor() == Triple.getVendor()) {
2009*12c85518Srobert       if (TargetVariantTriple)
2010*12c85518Srobert         continue;
2011*12c85518Srobert       A->claim();
2012*12c85518Srobert       // Accept a -target-variant triple when compiling code that may run on
2013*12c85518Srobert       // macOS or Mac Catalust.
2014*12c85518Srobert       if ((Triple.isMacOSX() && TVT.getOS() == llvm::Triple::IOS &&
2015*12c85518Srobert            TVT.isMacCatalystEnvironment()) ||
2016*12c85518Srobert           (TVT.isMacOSX() && Triple.getOS() == llvm::Triple::IOS &&
2017*12c85518Srobert            Triple.isMacCatalystEnvironment())) {
2018*12c85518Srobert         TargetVariantTriple = TVT;
2019*12c85518Srobert         continue;
2020*12c85518Srobert       }
2021*12c85518Srobert       TheDriver.Diag(diag::err_drv_target_variant_invalid)
2022*12c85518Srobert           << A->getSpelling() << A->getValue();
2023*12c85518Srobert     }
2024*12c85518Srobert   }
2025*12c85518Srobert   return DarwinPlatform::createFromTarget(Triple, OSVersion,
2026*12c85518Srobert                                           Args.getLastArg(options::OPT_target),
2027*12c85518Srobert                                           TargetVariantTriple, SDKInfo);
2028e5dd7070Spatrick }
2029e5dd7070Spatrick 
2030*12c85518Srobert /// Returns the deployment target that's specified using the -mtargetos option.
getDeploymentTargetFromMTargetOSArg(DerivedArgList & Args,const Driver & TheDriver,const std::optional<DarwinSDKInfo> & SDKInfo)2031*12c85518Srobert std::optional<DarwinPlatform> getDeploymentTargetFromMTargetOSArg(
2032*12c85518Srobert     DerivedArgList &Args, const Driver &TheDriver,
2033*12c85518Srobert     const std::optional<DarwinSDKInfo> &SDKInfo) {
2034*12c85518Srobert   auto *A = Args.getLastArg(options::OPT_mtargetos_EQ);
2035*12c85518Srobert   if (!A)
2036*12c85518Srobert     return std::nullopt;
2037*12c85518Srobert   llvm::Triple TT(llvm::Twine("unknown-apple-") + A->getValue());
2038*12c85518Srobert   switch (TT.getOS()) {
2039*12c85518Srobert   case llvm::Triple::MacOSX:
2040*12c85518Srobert   case llvm::Triple::IOS:
2041*12c85518Srobert   case llvm::Triple::TvOS:
2042*12c85518Srobert   case llvm::Triple::WatchOS:
2043*12c85518Srobert     break;
2044*12c85518Srobert   default:
2045*12c85518Srobert     TheDriver.Diag(diag::err_drv_invalid_os_in_arg)
2046*12c85518Srobert         << TT.getOSName() << A->getAsString(Args);
2047*12c85518Srobert     return std::nullopt;
2048*12c85518Srobert   }
2049*12c85518Srobert 
2050*12c85518Srobert   VersionTuple Version = TT.getOSVersion();
2051*12c85518Srobert   if (!Version.getMajor()) {
2052*12c85518Srobert     TheDriver.Diag(diag::err_drv_invalid_version_number)
2053*12c85518Srobert         << A->getAsString(Args);
2054*12c85518Srobert     return std::nullopt;
2055*12c85518Srobert   }
2056*12c85518Srobert   return DarwinPlatform::createFromMTargetOS(TT.getOS(), Version,
2057*12c85518Srobert                                              TT.getEnvironment(), A, SDKInfo);
2058*12c85518Srobert }
2059*12c85518Srobert 
parseSDKSettings(llvm::vfs::FileSystem & VFS,const ArgList & Args,const Driver & TheDriver)2060*12c85518Srobert std::optional<DarwinSDKInfo> parseSDKSettings(llvm::vfs::FileSystem &VFS,
2061e5dd7070Spatrick                                               const ArgList &Args,
2062e5dd7070Spatrick                                               const Driver &TheDriver) {
2063e5dd7070Spatrick   const Arg *A = Args.getLastArg(options::OPT_isysroot);
2064e5dd7070Spatrick   if (!A)
2065*12c85518Srobert     return std::nullopt;
2066e5dd7070Spatrick   StringRef isysroot = A->getValue();
2067a9ac8606Spatrick   auto SDKInfoOrErr = parseDarwinSDKInfo(VFS, isysroot);
2068e5dd7070Spatrick   if (!SDKInfoOrErr) {
2069e5dd7070Spatrick     llvm::consumeError(SDKInfoOrErr.takeError());
2070e5dd7070Spatrick     TheDriver.Diag(diag::warn_drv_darwin_sdk_invalid_settings);
2071*12c85518Srobert     return std::nullopt;
2072e5dd7070Spatrick   }
2073e5dd7070Spatrick   return *SDKInfoOrErr;
2074e5dd7070Spatrick }
2075e5dd7070Spatrick 
2076e5dd7070Spatrick } // namespace
2077e5dd7070Spatrick 
AddDeploymentTarget(DerivedArgList & Args) const2078e5dd7070Spatrick void Darwin::AddDeploymentTarget(DerivedArgList &Args) const {
2079e5dd7070Spatrick   const OptTable &Opts = getDriver().getOpts();
2080e5dd7070Spatrick 
2081e5dd7070Spatrick   // Support allowing the SDKROOT environment variable used by xcrun and other
2082e5dd7070Spatrick   // Xcode tools to define the default sysroot, by making it the default for
2083e5dd7070Spatrick   // isysroot.
2084e5dd7070Spatrick   if (const Arg *A = Args.getLastArg(options::OPT_isysroot)) {
2085e5dd7070Spatrick     // Warn if the path does not exist.
2086e5dd7070Spatrick     if (!getVFS().exists(A->getValue()))
2087e5dd7070Spatrick       getDriver().Diag(clang::diag::warn_missing_sysroot) << A->getValue();
2088e5dd7070Spatrick   } else {
2089e5dd7070Spatrick     if (char *env = ::getenv("SDKROOT")) {
2090e5dd7070Spatrick       // We only use this value as the default if it is an absolute path,
2091e5dd7070Spatrick       // exists, and it is not the root path.
2092e5dd7070Spatrick       if (llvm::sys::path::is_absolute(env) && getVFS().exists(env) &&
2093e5dd7070Spatrick           StringRef(env) != "/") {
2094e5dd7070Spatrick         Args.append(Args.MakeSeparateArg(
2095e5dd7070Spatrick             nullptr, Opts.getOption(options::OPT_isysroot), env));
2096e5dd7070Spatrick       }
2097e5dd7070Spatrick     }
2098e5dd7070Spatrick   }
2099e5dd7070Spatrick 
2100e5dd7070Spatrick   // Read the SDKSettings.json file for more information, like the SDK version
2101e5dd7070Spatrick   // that we can pass down to the compiler.
2102e5dd7070Spatrick   SDKInfo = parseSDKSettings(getVFS(), Args, getDriver());
2103e5dd7070Spatrick 
2104e5dd7070Spatrick   // The OS and the version can be specified using the -target argument.
2105*12c85518Srobert   std::optional<DarwinPlatform> OSTarget =
2106a9ac8606Spatrick       getDeploymentTargetFromTargetArg(Args, getTriple(), getDriver(), SDKInfo);
2107e5dd7070Spatrick   if (OSTarget) {
2108*12c85518Srobert     // Disallow mixing -target and -mtargetos=.
2109*12c85518Srobert     if (const auto *MTargetOSArg = Args.getLastArg(options::OPT_mtargetos_EQ)) {
2110*12c85518Srobert       std::string TargetArgStr = OSTarget->getAsString(Args, Opts);
2111*12c85518Srobert       std::string MTargetOSArgStr = MTargetOSArg->getAsString(Args);
2112*12c85518Srobert       getDriver().Diag(diag::err_drv_cannot_mix_options)
2113*12c85518Srobert           << TargetArgStr << MTargetOSArgStr;
2114*12c85518Srobert     }
2115*12c85518Srobert     std::optional<DarwinPlatform> OSVersionArgTarget =
2116e5dd7070Spatrick         getDeploymentTargetFromOSVersionArg(Args, getDriver());
2117e5dd7070Spatrick     if (OSVersionArgTarget) {
2118e5dd7070Spatrick       unsigned TargetMajor, TargetMinor, TargetMicro;
2119e5dd7070Spatrick       bool TargetExtra;
2120e5dd7070Spatrick       unsigned ArgMajor, ArgMinor, ArgMicro;
2121e5dd7070Spatrick       bool ArgExtra;
2122e5dd7070Spatrick       if (OSTarget->getPlatform() != OSVersionArgTarget->getPlatform() ||
2123e5dd7070Spatrick           (Driver::GetReleaseVersion(OSTarget->getOSVersion(), TargetMajor,
2124e5dd7070Spatrick                                      TargetMinor, TargetMicro, TargetExtra) &&
2125e5dd7070Spatrick            Driver::GetReleaseVersion(OSVersionArgTarget->getOSVersion(),
2126e5dd7070Spatrick                                      ArgMajor, ArgMinor, ArgMicro, ArgExtra) &&
2127e5dd7070Spatrick            (VersionTuple(TargetMajor, TargetMinor, TargetMicro) !=
2128e5dd7070Spatrick                 VersionTuple(ArgMajor, ArgMinor, ArgMicro) ||
2129e5dd7070Spatrick             TargetExtra != ArgExtra))) {
2130e5dd7070Spatrick         // Select the OS version from the -m<os>-version-min argument when
2131e5dd7070Spatrick         // the -target does not include an OS version.
2132e5dd7070Spatrick         if (OSTarget->getPlatform() == OSVersionArgTarget->getPlatform() &&
2133e5dd7070Spatrick             !OSTarget->hasOSVersion()) {
2134e5dd7070Spatrick           OSTarget->setOSVersion(OSVersionArgTarget->getOSVersion());
2135e5dd7070Spatrick         } else {
2136e5dd7070Spatrick           // Warn about -m<os>-version-min that doesn't match the OS version
2137e5dd7070Spatrick           // that's specified in the target.
2138e5dd7070Spatrick           std::string OSVersionArg =
2139e5dd7070Spatrick               OSVersionArgTarget->getAsString(Args, Opts);
2140e5dd7070Spatrick           std::string TargetArg = OSTarget->getAsString(Args, Opts);
2141e5dd7070Spatrick           getDriver().Diag(clang::diag::warn_drv_overriding_flag_option)
2142e5dd7070Spatrick               << OSVersionArg << TargetArg;
2143e5dd7070Spatrick         }
2144e5dd7070Spatrick       }
2145e5dd7070Spatrick     }
2146*12c85518Srobert   } else if ((OSTarget = getDeploymentTargetFromMTargetOSArg(Args, getDriver(),
2147*12c85518Srobert                                                              SDKInfo))) {
2148*12c85518Srobert     // The OS target can be specified using the -mtargetos= argument.
2149*12c85518Srobert     // Disallow mixing -mtargetos= and -m<os>version-min=.
2150*12c85518Srobert     std::optional<DarwinPlatform> OSVersionArgTarget =
2151*12c85518Srobert         getDeploymentTargetFromOSVersionArg(Args, getDriver());
2152*12c85518Srobert     if (OSVersionArgTarget) {
2153*12c85518Srobert       std::string MTargetOSArgStr = OSTarget->getAsString(Args, Opts);
2154*12c85518Srobert       std::string OSVersionArgStr = OSVersionArgTarget->getAsString(Args, Opts);
2155*12c85518Srobert       getDriver().Diag(diag::err_drv_cannot_mix_options)
2156*12c85518Srobert           << MTargetOSArgStr << OSVersionArgStr;
2157*12c85518Srobert     }
2158e5dd7070Spatrick   } else {
2159e5dd7070Spatrick     // The OS target can be specified using the -m<os>version-min argument.
2160e5dd7070Spatrick     OSTarget = getDeploymentTargetFromOSVersionArg(Args, getDriver());
2161e5dd7070Spatrick     // If no deployment target was specified on the command line, check for
2162e5dd7070Spatrick     // environment defines.
2163e5dd7070Spatrick     if (!OSTarget) {
2164e5dd7070Spatrick       OSTarget =
2165e5dd7070Spatrick           getDeploymentTargetFromEnvironmentVariables(getDriver(), getTriple());
2166e5dd7070Spatrick       if (OSTarget) {
2167e5dd7070Spatrick         // Don't infer simulator from the arch when the SDK is also specified.
2168*12c85518Srobert         std::optional<DarwinPlatform> SDKTarget =
2169e5dd7070Spatrick             inferDeploymentTargetFromSDK(Args, SDKInfo);
2170e5dd7070Spatrick         if (SDKTarget)
2171e5dd7070Spatrick           OSTarget->setEnvironment(SDKTarget->getEnvironment());
2172e5dd7070Spatrick       }
2173e5dd7070Spatrick     }
2174e5dd7070Spatrick     // If there is no command-line argument to specify the Target version and
2175e5dd7070Spatrick     // no environment variable defined, see if we can set the default based
2176e5dd7070Spatrick     // on -isysroot using SDKSettings.json if it exists.
2177e5dd7070Spatrick     if (!OSTarget) {
2178e5dd7070Spatrick       OSTarget = inferDeploymentTargetFromSDK(Args, SDKInfo);
2179e5dd7070Spatrick       /// If the target was successfully constructed from the SDK path, try to
2180e5dd7070Spatrick       /// infer the SDK info if the SDK doesn't have it.
2181e5dd7070Spatrick       if (OSTarget && !SDKInfo)
2182e5dd7070Spatrick         SDKInfo = OSTarget->inferSDKInfo();
2183e5dd7070Spatrick     }
2184e5dd7070Spatrick     // If no OS targets have been specified, try to guess platform from -target
2185e5dd7070Spatrick     // or arch name and compute the version from the triple.
2186e5dd7070Spatrick     if (!OSTarget)
2187e5dd7070Spatrick       OSTarget =
2188e5dd7070Spatrick           inferDeploymentTargetFromArch(Args, *this, getTriple(), getDriver());
2189e5dd7070Spatrick   }
2190e5dd7070Spatrick 
2191e5dd7070Spatrick   assert(OSTarget && "Unable to infer Darwin variant");
2192e5dd7070Spatrick   OSTarget->addOSVersionMinArgument(Args, Opts);
2193e5dd7070Spatrick   DarwinPlatformKind Platform = OSTarget->getPlatform();
2194e5dd7070Spatrick 
2195e5dd7070Spatrick   unsigned Major, Minor, Micro;
2196e5dd7070Spatrick   bool HadExtra;
2197*12c85518Srobert   // The major version should not be over this number.
2198*12c85518Srobert   const unsigned MajorVersionLimit = 1000;
2199e5dd7070Spatrick   // Set the tool chain target information.
2200e5dd7070Spatrick   if (Platform == MacOS) {
2201e5dd7070Spatrick     if (!Driver::GetReleaseVersion(OSTarget->getOSVersion(), Major, Minor,
2202e5dd7070Spatrick                                    Micro, HadExtra) ||
2203*12c85518Srobert         HadExtra || Major < 10 || Major >= MajorVersionLimit || Minor >= 100 ||
2204*12c85518Srobert         Micro >= 100)
2205e5dd7070Spatrick       getDriver().Diag(diag::err_drv_invalid_version_number)
2206e5dd7070Spatrick           << OSTarget->getAsString(Args, Opts);
2207e5dd7070Spatrick   } else if (Platform == IPhoneOS) {
2208e5dd7070Spatrick     if (!Driver::GetReleaseVersion(OSTarget->getOSVersion(), Major, Minor,
2209e5dd7070Spatrick                                    Micro, HadExtra) ||
2210*12c85518Srobert         HadExtra || Major >= MajorVersionLimit || Minor >= 100 || Micro >= 100)
2211e5dd7070Spatrick       getDriver().Diag(diag::err_drv_invalid_version_number)
2212e5dd7070Spatrick           << OSTarget->getAsString(Args, Opts);
2213e5dd7070Spatrick     ;
2214a9ac8606Spatrick     if (OSTarget->getEnvironment() == MacCatalyst &&
2215a9ac8606Spatrick         (Major < 13 || (Major == 13 && Minor < 1))) {
2216a9ac8606Spatrick       getDriver().Diag(diag::err_drv_invalid_version_number)
2217a9ac8606Spatrick           << OSTarget->getAsString(Args, Opts);
2218a9ac8606Spatrick       Major = 13;
2219a9ac8606Spatrick       Minor = 1;
2220a9ac8606Spatrick       Micro = 0;
2221a9ac8606Spatrick     }
2222e5dd7070Spatrick     // For 32-bit targets, the deployment target for iOS has to be earlier than
2223e5dd7070Spatrick     // iOS 11.
2224e5dd7070Spatrick     if (getTriple().isArch32Bit() && Major >= 11) {
2225e5dd7070Spatrick       // If the deployment target is explicitly specified, print a diagnostic.
2226e5dd7070Spatrick       if (OSTarget->isExplicitlySpecified()) {
2227a9ac8606Spatrick         if (OSTarget->getEnvironment() == MacCatalyst)
2228a9ac8606Spatrick           getDriver().Diag(diag::err_invalid_macos_32bit_deployment_target);
2229a9ac8606Spatrick         else
2230e5dd7070Spatrick           getDriver().Diag(diag::warn_invalid_ios_deployment_target)
2231e5dd7070Spatrick               << OSTarget->getAsString(Args, Opts);
2232e5dd7070Spatrick         // Otherwise, set it to 10.99.99.
2233e5dd7070Spatrick       } else {
2234e5dd7070Spatrick         Major = 10;
2235e5dd7070Spatrick         Minor = 99;
2236e5dd7070Spatrick         Micro = 99;
2237e5dd7070Spatrick       }
2238e5dd7070Spatrick     }
2239e5dd7070Spatrick   } else if (Platform == TvOS) {
2240e5dd7070Spatrick     if (!Driver::GetReleaseVersion(OSTarget->getOSVersion(), Major, Minor,
2241e5dd7070Spatrick                                    Micro, HadExtra) ||
2242*12c85518Srobert         HadExtra || Major >= MajorVersionLimit || Minor >= 100 || Micro >= 100)
2243e5dd7070Spatrick       getDriver().Diag(diag::err_drv_invalid_version_number)
2244e5dd7070Spatrick           << OSTarget->getAsString(Args, Opts);
2245e5dd7070Spatrick   } else if (Platform == WatchOS) {
2246e5dd7070Spatrick     if (!Driver::GetReleaseVersion(OSTarget->getOSVersion(), Major, Minor,
2247e5dd7070Spatrick                                    Micro, HadExtra) ||
2248*12c85518Srobert         HadExtra || Major >= MajorVersionLimit || Minor >= 100 || Micro >= 100)
2249*12c85518Srobert       getDriver().Diag(diag::err_drv_invalid_version_number)
2250*12c85518Srobert           << OSTarget->getAsString(Args, Opts);
2251*12c85518Srobert   } else if (Platform == DriverKit) {
2252*12c85518Srobert     if (!Driver::GetReleaseVersion(OSTarget->getOSVersion(), Major, Minor,
2253*12c85518Srobert                                    Micro, HadExtra) ||
2254*12c85518Srobert         HadExtra || Major < 19 || Major >= MajorVersionLimit || Minor >= 100 ||
2255*12c85518Srobert         Micro >= 100)
2256e5dd7070Spatrick       getDriver().Diag(diag::err_drv_invalid_version_number)
2257e5dd7070Spatrick           << OSTarget->getAsString(Args, Opts);
2258e5dd7070Spatrick   } else
2259e5dd7070Spatrick     llvm_unreachable("unknown kind of Darwin platform");
2260e5dd7070Spatrick 
2261e5dd7070Spatrick   DarwinEnvironmentKind Environment = OSTarget->getEnvironment();
2262e5dd7070Spatrick   // Recognize iOS targets with an x86 architecture as the iOS simulator.
2263e5dd7070Spatrick   if (Environment == NativeEnvironment && Platform != MacOS &&
2264*12c85518Srobert       Platform != DriverKit && OSTarget->canInferSimulatorFromArch() &&
2265*12c85518Srobert       getTriple().isX86())
2266e5dd7070Spatrick     Environment = Simulator;
2267e5dd7070Spatrick 
2268a9ac8606Spatrick   VersionTuple NativeTargetVersion;
2269a9ac8606Spatrick   if (Environment == MacCatalyst)
2270a9ac8606Spatrick     NativeTargetVersion = OSTarget->getNativeTargetVersion();
2271a9ac8606Spatrick   setTarget(Platform, Environment, Major, Minor, Micro, NativeTargetVersion);
2272*12c85518Srobert   TargetVariantTriple = OSTarget->getTargetVariantTriple();
2273e5dd7070Spatrick 
2274e5dd7070Spatrick   if (const Arg *A = Args.getLastArg(options::OPT_isysroot)) {
2275e5dd7070Spatrick     StringRef SDK = getSDKName(A->getValue());
2276e5dd7070Spatrick     if (SDK.size() > 0) {
2277e5dd7070Spatrick       size_t StartVer = SDK.find_first_of("0123456789");
2278e5dd7070Spatrick       StringRef SDKName = SDK.slice(0, StartVer);
2279a9ac8606Spatrick       if (!SDKName.startswith(getPlatformFamily()) &&
2280a9ac8606Spatrick           !dropSDKNamePrefix(SDKName).startswith(getPlatformFamily()))
2281e5dd7070Spatrick         getDriver().Diag(diag::warn_incompatible_sysroot)
2282e5dd7070Spatrick             << SDKName << getPlatformFamily();
2283e5dd7070Spatrick     }
2284e5dd7070Spatrick   }
2285e5dd7070Spatrick }
2286e5dd7070Spatrick 
2287*12c85518Srobert // For certain platforms/environments almost all resources (e.g., headers) are
2288*12c85518Srobert // located in sub-directories, e.g., for DriverKit they live in
2289*12c85518Srobert // <SYSROOT>/System/DriverKit/usr/include (instead of <SYSROOT>/usr/include).
AppendPlatformPrefix(SmallString<128> & Path,const llvm::Triple & T)2290*12c85518Srobert static void AppendPlatformPrefix(SmallString<128> &Path,
2291*12c85518Srobert                                  const llvm::Triple &T) {
2292*12c85518Srobert   if (T.isDriverKit()) {
2293*12c85518Srobert     llvm::sys::path::append(Path, "System", "DriverKit");
2294*12c85518Srobert   }
2295*12c85518Srobert }
2296*12c85518Srobert 
2297*12c85518Srobert // Returns the effective sysroot from either -isysroot or --sysroot, plus the
2298*12c85518Srobert // platform prefix (if any).
2299*12c85518Srobert llvm::SmallString<128>
GetEffectiveSysroot(const llvm::opt::ArgList & DriverArgs) const2300*12c85518Srobert DarwinClang::GetEffectiveSysroot(const llvm::opt::ArgList &DriverArgs) const {
2301*12c85518Srobert   llvm::SmallString<128> Path("/");
2302e5dd7070Spatrick   if (DriverArgs.hasArg(options::OPT_isysroot))
2303*12c85518Srobert     Path = DriverArgs.getLastArgValue(options::OPT_isysroot);
2304*12c85518Srobert   else if (!getDriver().SysRoot.empty())
2305*12c85518Srobert     Path = getDriver().SysRoot;
2306*12c85518Srobert 
2307*12c85518Srobert   if (hasEffectiveTriple()) {
2308*12c85518Srobert     AppendPlatformPrefix(Path, getEffectiveTriple());
2309*12c85518Srobert   }
2310*12c85518Srobert   return Path;
2311e5dd7070Spatrick }
2312e5dd7070Spatrick 
AddClangSystemIncludeArgs(const llvm::opt::ArgList & DriverArgs,llvm::opt::ArgStringList & CC1Args) const2313e5dd7070Spatrick void DarwinClang::AddClangSystemIncludeArgs(const llvm::opt::ArgList &DriverArgs,
2314e5dd7070Spatrick                                             llvm::opt::ArgStringList &CC1Args) const {
2315e5dd7070Spatrick   const Driver &D = getDriver();
2316e5dd7070Spatrick 
2317*12c85518Srobert   llvm::SmallString<128> Sysroot = GetEffectiveSysroot(DriverArgs);
2318e5dd7070Spatrick 
2319e5dd7070Spatrick   bool NoStdInc = DriverArgs.hasArg(options::OPT_nostdinc);
2320e5dd7070Spatrick   bool NoStdlibInc = DriverArgs.hasArg(options::OPT_nostdlibinc);
2321ec727ea7Spatrick   bool NoBuiltinInc = DriverArgs.hasFlag(
2322ec727ea7Spatrick       options::OPT_nobuiltininc, options::OPT_ibuiltininc, /*Default=*/false);
2323ec727ea7Spatrick   bool ForceBuiltinInc = DriverArgs.hasFlag(
2324ec727ea7Spatrick       options::OPT_ibuiltininc, options::OPT_nobuiltininc, /*Default=*/false);
2325e5dd7070Spatrick 
2326e5dd7070Spatrick   // Add <sysroot>/usr/local/include
2327e5dd7070Spatrick   if (!NoStdInc && !NoStdlibInc) {
2328e5dd7070Spatrick       SmallString<128> P(Sysroot);
2329e5dd7070Spatrick       llvm::sys::path::append(P, "usr", "local", "include");
2330e5dd7070Spatrick       addSystemInclude(DriverArgs, CC1Args, P);
2331e5dd7070Spatrick   }
2332e5dd7070Spatrick 
2333e5dd7070Spatrick   // Add the Clang builtin headers (<resource>/include)
2334ec727ea7Spatrick   if (!(NoStdInc && !ForceBuiltinInc) && !NoBuiltinInc) {
2335e5dd7070Spatrick     SmallString<128> P(D.ResourceDir);
2336e5dd7070Spatrick     llvm::sys::path::append(P, "include");
2337e5dd7070Spatrick     addSystemInclude(DriverArgs, CC1Args, P);
2338e5dd7070Spatrick   }
2339e5dd7070Spatrick 
2340e5dd7070Spatrick   if (NoStdInc || NoStdlibInc)
2341e5dd7070Spatrick     return;
2342e5dd7070Spatrick 
2343e5dd7070Spatrick   // Check for configure-time C include directories.
2344e5dd7070Spatrick   llvm::StringRef CIncludeDirs(C_INCLUDE_DIRS);
2345e5dd7070Spatrick   if (!CIncludeDirs.empty()) {
2346e5dd7070Spatrick     llvm::SmallVector<llvm::StringRef, 5> dirs;
2347e5dd7070Spatrick     CIncludeDirs.split(dirs, ":");
2348e5dd7070Spatrick     for (llvm::StringRef dir : dirs) {
2349e5dd7070Spatrick       llvm::StringRef Prefix =
2350ec727ea7Spatrick           llvm::sys::path::is_absolute(dir) ? "" : llvm::StringRef(Sysroot);
2351e5dd7070Spatrick       addExternCSystemInclude(DriverArgs, CC1Args, Prefix + dir);
2352e5dd7070Spatrick     }
2353e5dd7070Spatrick   } else {
2354e5dd7070Spatrick     // Otherwise, add <sysroot>/usr/include.
2355e5dd7070Spatrick     SmallString<128> P(Sysroot);
2356e5dd7070Spatrick     llvm::sys::path::append(P, "usr", "include");
2357e5dd7070Spatrick     addExternCSystemInclude(DriverArgs, CC1Args, P.str());
2358e5dd7070Spatrick   }
2359e5dd7070Spatrick }
2360e5dd7070Spatrick 
AddGnuCPlusPlusIncludePaths(const llvm::opt::ArgList & DriverArgs,llvm::opt::ArgStringList & CC1Args,llvm::SmallString<128> Base,llvm::StringRef Version,llvm::StringRef ArchDir,llvm::StringRef BitDir) const2361e5dd7070Spatrick bool DarwinClang::AddGnuCPlusPlusIncludePaths(const llvm::opt::ArgList &DriverArgs,
2362e5dd7070Spatrick                                               llvm::opt::ArgStringList &CC1Args,
2363e5dd7070Spatrick                                               llvm::SmallString<128> Base,
2364e5dd7070Spatrick                                               llvm::StringRef Version,
2365e5dd7070Spatrick                                               llvm::StringRef ArchDir,
2366e5dd7070Spatrick                                               llvm::StringRef BitDir) const {
2367e5dd7070Spatrick   llvm::sys::path::append(Base, Version);
2368e5dd7070Spatrick 
2369e5dd7070Spatrick   // Add the base dir
2370e5dd7070Spatrick   addSystemInclude(DriverArgs, CC1Args, Base);
2371e5dd7070Spatrick 
2372e5dd7070Spatrick   // Add the multilib dirs
2373e5dd7070Spatrick   {
2374e5dd7070Spatrick     llvm::SmallString<128> P = Base;
2375e5dd7070Spatrick     if (!ArchDir.empty())
2376e5dd7070Spatrick       llvm::sys::path::append(P, ArchDir);
2377e5dd7070Spatrick     if (!BitDir.empty())
2378e5dd7070Spatrick       llvm::sys::path::append(P, BitDir);
2379e5dd7070Spatrick     addSystemInclude(DriverArgs, CC1Args, P);
2380e5dd7070Spatrick   }
2381e5dd7070Spatrick 
2382e5dd7070Spatrick   // Add the backward dir
2383e5dd7070Spatrick   {
2384e5dd7070Spatrick     llvm::SmallString<128> P = Base;
2385e5dd7070Spatrick     llvm::sys::path::append(P, "backward");
2386e5dd7070Spatrick     addSystemInclude(DriverArgs, CC1Args, P);
2387e5dd7070Spatrick   }
2388e5dd7070Spatrick 
2389e5dd7070Spatrick   return getVFS().exists(Base);
2390e5dd7070Spatrick }
2391e5dd7070Spatrick 
AddClangCXXStdlibIncludeArgs(const llvm::opt::ArgList & DriverArgs,llvm::opt::ArgStringList & CC1Args) const2392e5dd7070Spatrick void DarwinClang::AddClangCXXStdlibIncludeArgs(
2393e5dd7070Spatrick     const llvm::opt::ArgList &DriverArgs,
2394e5dd7070Spatrick     llvm::opt::ArgStringList &CC1Args) const {
2395e5dd7070Spatrick   // The implementation from a base class will pass through the -stdlib to
2396e5dd7070Spatrick   // CC1Args.
2397e5dd7070Spatrick   // FIXME: this should not be necessary, remove usages in the frontend
2398e5dd7070Spatrick   //        (e.g. HeaderSearchOptions::UseLibcxx) and don't pipe -stdlib.
2399e5dd7070Spatrick   //        Also check whether this is used for setting library search paths.
2400e5dd7070Spatrick   ToolChain::AddClangCXXStdlibIncludeArgs(DriverArgs, CC1Args);
2401e5dd7070Spatrick 
2402*12c85518Srobert   if (DriverArgs.hasArg(options::OPT_nostdinc, options::OPT_nostdlibinc,
2403*12c85518Srobert                         options::OPT_nostdincxx))
2404e5dd7070Spatrick     return;
2405e5dd7070Spatrick 
2406*12c85518Srobert   llvm::SmallString<128> Sysroot = GetEffectiveSysroot(DriverArgs);
2407e5dd7070Spatrick 
2408e5dd7070Spatrick   switch (GetCXXStdlibType(DriverArgs)) {
2409e5dd7070Spatrick   case ToolChain::CST_Libcxx: {
2410a9ac8606Spatrick     // On Darwin, libc++ can be installed in one of the following two places:
2411a9ac8606Spatrick     // 1. Alongside the compiler in         <install>/include/c++/v1
2412a9ac8606Spatrick     // 2. In a SDK (or a custom sysroot) in <sysroot>/usr/include/c++/v1
2413a9ac8606Spatrick     //
2414a9ac8606Spatrick     // The precendence of paths is as listed above, i.e. we take the first path
2415a9ac8606Spatrick     // that exists. Also note that we never include libc++ twice -- we take the
2416a9ac8606Spatrick     // first path that exists and don't send the other paths to CC1 (otherwise
2417a9ac8606Spatrick     // include_next could break).
2418a9ac8606Spatrick 
2419a9ac8606Spatrick     // Check for (1)
2420a9ac8606Spatrick     // Get from '<install>/bin' to '<install>/include/c++/v1'.
2421a9ac8606Spatrick     // Note that InstallBin can be relative, so we use '..' instead of
2422a9ac8606Spatrick     // parent_path.
2423a9ac8606Spatrick     llvm::SmallString<128> InstallBin =
2424a9ac8606Spatrick         llvm::StringRef(getDriver().getInstalledDir()); // <install>/bin
2425a9ac8606Spatrick     llvm::sys::path::append(InstallBin, "..", "include", "c++", "v1");
2426a9ac8606Spatrick     if (getVFS().exists(InstallBin)) {
2427a9ac8606Spatrick       addSystemInclude(DriverArgs, CC1Args, InstallBin);
2428a9ac8606Spatrick       return;
2429a9ac8606Spatrick     } else if (DriverArgs.hasArg(options::OPT_v)) {
2430a9ac8606Spatrick       llvm::errs() << "ignoring nonexistent directory \"" << InstallBin
2431a9ac8606Spatrick                    << "\"\n";
2432e5dd7070Spatrick     }
2433a9ac8606Spatrick 
2434a9ac8606Spatrick     // Otherwise, check for (2)
2435a9ac8606Spatrick     llvm::SmallString<128> SysrootUsr = Sysroot;
2436a9ac8606Spatrick     llvm::sys::path::append(SysrootUsr, "usr", "include", "c++", "v1");
2437a9ac8606Spatrick     if (getVFS().exists(SysrootUsr)) {
2438a9ac8606Spatrick       addSystemInclude(DriverArgs, CC1Args, SysrootUsr);
2439a9ac8606Spatrick       return;
2440a9ac8606Spatrick     } else if (DriverArgs.hasArg(options::OPT_v)) {
2441a9ac8606Spatrick       llvm::errs() << "ignoring nonexistent directory \"" << SysrootUsr
2442a9ac8606Spatrick                    << "\"\n";
2443e5dd7070Spatrick     }
2444a9ac8606Spatrick 
2445a9ac8606Spatrick     // Otherwise, don't add any path.
2446e5dd7070Spatrick     break;
2447e5dd7070Spatrick   }
2448e5dd7070Spatrick 
2449e5dd7070Spatrick   case ToolChain::CST_Libstdcxx:
2450e5dd7070Spatrick     llvm::SmallString<128> UsrIncludeCxx = Sysroot;
2451e5dd7070Spatrick     llvm::sys::path::append(UsrIncludeCxx, "usr", "include", "c++");
2452e5dd7070Spatrick 
2453e5dd7070Spatrick     llvm::Triple::ArchType arch = getTriple().getArch();
2454e5dd7070Spatrick     bool IsBaseFound = true;
2455e5dd7070Spatrick     switch (arch) {
2456e5dd7070Spatrick     default: break;
2457e5dd7070Spatrick 
2458e5dd7070Spatrick     case llvm::Triple::ppc:
2459e5dd7070Spatrick     case llvm::Triple::ppc64:
2460e5dd7070Spatrick       IsBaseFound = AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2461e5dd7070Spatrick                                                 "4.2.1",
2462e5dd7070Spatrick                                                 "powerpc-apple-darwin10",
2463e5dd7070Spatrick                                                 arch == llvm::Triple::ppc64 ? "ppc64" : "");
2464e5dd7070Spatrick       IsBaseFound |= AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2465e5dd7070Spatrick                                                 "4.0.0", "powerpc-apple-darwin10",
2466e5dd7070Spatrick                                                  arch == llvm::Triple::ppc64 ? "ppc64" : "");
2467e5dd7070Spatrick       break;
2468e5dd7070Spatrick 
2469e5dd7070Spatrick     case llvm::Triple::x86:
2470e5dd7070Spatrick     case llvm::Triple::x86_64:
2471e5dd7070Spatrick       IsBaseFound = AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2472e5dd7070Spatrick                                                 "4.2.1",
2473e5dd7070Spatrick                                                 "i686-apple-darwin10",
2474e5dd7070Spatrick                                                 arch == llvm::Triple::x86_64 ? "x86_64" : "");
2475e5dd7070Spatrick       IsBaseFound |= AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2476e5dd7070Spatrick                                                 "4.0.0", "i686-apple-darwin8",
2477e5dd7070Spatrick                                                  "");
2478e5dd7070Spatrick       break;
2479e5dd7070Spatrick 
2480e5dd7070Spatrick     case llvm::Triple::arm:
2481e5dd7070Spatrick     case llvm::Triple::thumb:
2482e5dd7070Spatrick       IsBaseFound = AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2483e5dd7070Spatrick                                                 "4.2.1",
2484e5dd7070Spatrick                                                 "arm-apple-darwin10",
2485e5dd7070Spatrick                                                 "v7");
2486e5dd7070Spatrick       IsBaseFound |= AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2487e5dd7070Spatrick                                                 "4.2.1",
2488e5dd7070Spatrick                                                 "arm-apple-darwin10",
2489e5dd7070Spatrick                                                  "v6");
2490e5dd7070Spatrick       break;
2491e5dd7070Spatrick 
2492e5dd7070Spatrick     case llvm::Triple::aarch64:
2493e5dd7070Spatrick       IsBaseFound = AddGnuCPlusPlusIncludePaths(DriverArgs, CC1Args, UsrIncludeCxx,
2494e5dd7070Spatrick                                                 "4.2.1",
2495e5dd7070Spatrick                                                 "arm64-apple-darwin10",
2496e5dd7070Spatrick                                                 "");
2497e5dd7070Spatrick       break;
2498e5dd7070Spatrick     }
2499e5dd7070Spatrick 
2500e5dd7070Spatrick     if (!IsBaseFound) {
2501e5dd7070Spatrick       getDriver().Diag(diag::warn_drv_libstdcxx_not_found);
2502e5dd7070Spatrick     }
2503e5dd7070Spatrick 
2504e5dd7070Spatrick     break;
2505e5dd7070Spatrick   }
2506e5dd7070Spatrick }
2507*12c85518Srobert 
AddCXXStdlibLibArgs(const ArgList & Args,ArgStringList & CmdArgs) const2508e5dd7070Spatrick void DarwinClang::AddCXXStdlibLibArgs(const ArgList &Args,
2509e5dd7070Spatrick                                       ArgStringList &CmdArgs) const {
2510e5dd7070Spatrick   CXXStdlibType Type = GetCXXStdlibType(Args);
2511e5dd7070Spatrick 
2512e5dd7070Spatrick   switch (Type) {
2513e5dd7070Spatrick   case ToolChain::CST_Libcxx:
2514e5dd7070Spatrick     CmdArgs.push_back("-lc++");
2515*12c85518Srobert     if (Args.hasArg(options::OPT_fexperimental_library))
2516*12c85518Srobert       CmdArgs.push_back("-lc++experimental");
2517e5dd7070Spatrick     break;
2518e5dd7070Spatrick 
2519e5dd7070Spatrick   case ToolChain::CST_Libstdcxx:
2520e5dd7070Spatrick     // Unfortunately, -lstdc++ doesn't always exist in the standard search path;
2521e5dd7070Spatrick     // it was previously found in the gcc lib dir. However, for all the Darwin
2522e5dd7070Spatrick     // platforms we care about it was -lstdc++.6, so we search for that
2523e5dd7070Spatrick     // explicitly if we can't see an obvious -lstdc++ candidate.
2524e5dd7070Spatrick 
2525e5dd7070Spatrick     // Check in the sysroot first.
2526e5dd7070Spatrick     if (const Arg *A = Args.getLastArg(options::OPT_isysroot)) {
2527e5dd7070Spatrick       SmallString<128> P(A->getValue());
2528e5dd7070Spatrick       llvm::sys::path::append(P, "usr", "lib", "libstdc++.dylib");
2529e5dd7070Spatrick 
2530e5dd7070Spatrick       if (!getVFS().exists(P)) {
2531e5dd7070Spatrick         llvm::sys::path::remove_filename(P);
2532e5dd7070Spatrick         llvm::sys::path::append(P, "libstdc++.6.dylib");
2533e5dd7070Spatrick         if (getVFS().exists(P)) {
2534e5dd7070Spatrick           CmdArgs.push_back(Args.MakeArgString(P));
2535e5dd7070Spatrick           return;
2536e5dd7070Spatrick         }
2537e5dd7070Spatrick       }
2538e5dd7070Spatrick     }
2539e5dd7070Spatrick 
2540e5dd7070Spatrick     // Otherwise, look in the root.
2541e5dd7070Spatrick     // FIXME: This should be removed someday when we don't have to care about
2542e5dd7070Spatrick     // 10.6 and earlier, where /usr/lib/libstdc++.dylib does not exist.
2543e5dd7070Spatrick     if (!getVFS().exists("/usr/lib/libstdc++.dylib") &&
2544e5dd7070Spatrick         getVFS().exists("/usr/lib/libstdc++.6.dylib")) {
2545e5dd7070Spatrick       CmdArgs.push_back("/usr/lib/libstdc++.6.dylib");
2546e5dd7070Spatrick       return;
2547e5dd7070Spatrick     }
2548e5dd7070Spatrick 
2549e5dd7070Spatrick     // Otherwise, let the linker search.
2550e5dd7070Spatrick     CmdArgs.push_back("-lstdc++");
2551e5dd7070Spatrick     break;
2552e5dd7070Spatrick   }
2553e5dd7070Spatrick }
2554e5dd7070Spatrick 
AddCCKextLibArgs(const ArgList & Args,ArgStringList & CmdArgs) const2555e5dd7070Spatrick void DarwinClang::AddCCKextLibArgs(const ArgList &Args,
2556e5dd7070Spatrick                                    ArgStringList &CmdArgs) const {
2557e5dd7070Spatrick   // For Darwin platforms, use the compiler-rt-based support library
2558e5dd7070Spatrick   // instead of the gcc-provided one (which is also incidentally
2559e5dd7070Spatrick   // only present in the gcc lib dir, which makes it hard to find).
2560e5dd7070Spatrick 
2561e5dd7070Spatrick   SmallString<128> P(getDriver().ResourceDir);
2562e5dd7070Spatrick   llvm::sys::path::append(P, "lib", "darwin");
2563e5dd7070Spatrick 
2564e5dd7070Spatrick   // Use the newer cc_kext for iOS ARM after 6.0.
2565e5dd7070Spatrick   if (isTargetWatchOS()) {
2566e5dd7070Spatrick     llvm::sys::path::append(P, "libclang_rt.cc_kext_watchos.a");
2567e5dd7070Spatrick   } else if (isTargetTvOS()) {
2568e5dd7070Spatrick     llvm::sys::path::append(P, "libclang_rt.cc_kext_tvos.a");
2569e5dd7070Spatrick   } else if (isTargetIPhoneOS()) {
2570e5dd7070Spatrick     llvm::sys::path::append(P, "libclang_rt.cc_kext_ios.a");
2571*12c85518Srobert   } else if (isTargetDriverKit()) {
2572*12c85518Srobert     // DriverKit doesn't want extra runtime support.
2573e5dd7070Spatrick   } else {
2574e5dd7070Spatrick     llvm::sys::path::append(P, "libclang_rt.cc_kext.a");
2575e5dd7070Spatrick   }
2576e5dd7070Spatrick 
2577e5dd7070Spatrick   // For now, allow missing resource libraries to support developers who may
2578e5dd7070Spatrick   // not have compiler-rt checked out or integrated into their build.
2579e5dd7070Spatrick   if (getVFS().exists(P))
2580e5dd7070Spatrick     CmdArgs.push_back(Args.MakeArgString(P));
2581e5dd7070Spatrick }
2582e5dd7070Spatrick 
TranslateArgs(const DerivedArgList & Args,StringRef BoundArch,Action::OffloadKind) const2583e5dd7070Spatrick DerivedArgList *MachO::TranslateArgs(const DerivedArgList &Args,
2584e5dd7070Spatrick                                      StringRef BoundArch,
2585e5dd7070Spatrick                                      Action::OffloadKind) const {
2586e5dd7070Spatrick   DerivedArgList *DAL = new DerivedArgList(Args.getBaseArgs());
2587e5dd7070Spatrick   const OptTable &Opts = getDriver().getOpts();
2588e5dd7070Spatrick 
2589e5dd7070Spatrick   // FIXME: We really want to get out of the tool chain level argument
2590e5dd7070Spatrick   // translation business, as it makes the driver functionality much
2591e5dd7070Spatrick   // more opaque. For now, we follow gcc closely solely for the
2592e5dd7070Spatrick   // purpose of easily achieving feature parity & testability. Once we
2593e5dd7070Spatrick   // have something that works, we should reevaluate each translation
2594e5dd7070Spatrick   // and try to push it down into tool specific logic.
2595e5dd7070Spatrick 
2596e5dd7070Spatrick   for (Arg *A : Args) {
2597e5dd7070Spatrick     if (A->getOption().matches(options::OPT_Xarch__)) {
2598e5dd7070Spatrick       // Skip this argument unless the architecture matches either the toolchain
2599e5dd7070Spatrick       // triple arch, or the arch being bound.
2600*12c85518Srobert       StringRef XarchArch = A->getValue(0);
2601*12c85518Srobert       if (!(XarchArch == getArchName() ||
2602*12c85518Srobert             (!BoundArch.empty() && XarchArch == BoundArch)))
2603e5dd7070Spatrick         continue;
2604e5dd7070Spatrick 
2605e5dd7070Spatrick       Arg *OriginalArg = A;
2606ec727ea7Spatrick       TranslateXarchArgs(Args, A, DAL);
2607e5dd7070Spatrick 
2608e5dd7070Spatrick       // Linker input arguments require custom handling. The problem is that we
2609e5dd7070Spatrick       // have already constructed the phase actions, so we can not treat them as
2610e5dd7070Spatrick       // "input arguments".
2611e5dd7070Spatrick       if (A->getOption().hasFlag(options::LinkerInput)) {
2612e5dd7070Spatrick         // Convert the argument into individual Zlinker_input_args.
2613e5dd7070Spatrick         for (const char *Value : A->getValues()) {
2614e5dd7070Spatrick           DAL->AddSeparateArg(
2615e5dd7070Spatrick               OriginalArg, Opts.getOption(options::OPT_Zlinker_input), Value);
2616e5dd7070Spatrick         }
2617e5dd7070Spatrick         continue;
2618e5dd7070Spatrick       }
2619e5dd7070Spatrick     }
2620e5dd7070Spatrick 
2621e5dd7070Spatrick     // Sob. These is strictly gcc compatible for the time being. Apple
2622e5dd7070Spatrick     // gcc translates options twice, which means that self-expanding
2623e5dd7070Spatrick     // options add duplicates.
2624e5dd7070Spatrick     switch ((options::ID)A->getOption().getID()) {
2625e5dd7070Spatrick     default:
2626e5dd7070Spatrick       DAL->append(A);
2627e5dd7070Spatrick       break;
2628e5dd7070Spatrick 
2629e5dd7070Spatrick     case options::OPT_mkernel:
2630e5dd7070Spatrick     case options::OPT_fapple_kext:
2631e5dd7070Spatrick       DAL->append(A);
2632e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_static));
2633e5dd7070Spatrick       break;
2634e5dd7070Spatrick 
2635e5dd7070Spatrick     case options::OPT_dependency_file:
2636e5dd7070Spatrick       DAL->AddSeparateArg(A, Opts.getOption(options::OPT_MF), A->getValue());
2637e5dd7070Spatrick       break;
2638e5dd7070Spatrick 
2639e5dd7070Spatrick     case options::OPT_gfull:
2640e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_g_Flag));
2641e5dd7070Spatrick       DAL->AddFlagArg(
2642e5dd7070Spatrick           A, Opts.getOption(options::OPT_fno_eliminate_unused_debug_symbols));
2643e5dd7070Spatrick       break;
2644e5dd7070Spatrick 
2645e5dd7070Spatrick     case options::OPT_gused:
2646e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_g_Flag));
2647e5dd7070Spatrick       DAL->AddFlagArg(
2648e5dd7070Spatrick           A, Opts.getOption(options::OPT_feliminate_unused_debug_symbols));
2649e5dd7070Spatrick       break;
2650e5dd7070Spatrick 
2651e5dd7070Spatrick     case options::OPT_shared:
2652e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_dynamiclib));
2653e5dd7070Spatrick       break;
2654e5dd7070Spatrick 
2655e5dd7070Spatrick     case options::OPT_fconstant_cfstrings:
2656e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_mconstant_cfstrings));
2657e5dd7070Spatrick       break;
2658e5dd7070Spatrick 
2659e5dd7070Spatrick     case options::OPT_fno_constant_cfstrings:
2660e5dd7070Spatrick       DAL->AddFlagArg(A, Opts.getOption(options::OPT_mno_constant_cfstrings));
2661e5dd7070Spatrick       break;
2662e5dd7070Spatrick 
2663e5dd7070Spatrick     case options::OPT_Wnonportable_cfstrings:
2664e5dd7070Spatrick       DAL->AddFlagArg(A,
2665e5dd7070Spatrick                       Opts.getOption(options::OPT_mwarn_nonportable_cfstrings));
2666e5dd7070Spatrick       break;
2667e5dd7070Spatrick 
2668e5dd7070Spatrick     case options::OPT_Wno_nonportable_cfstrings:
2669e5dd7070Spatrick       DAL->AddFlagArg(
2670e5dd7070Spatrick           A, Opts.getOption(options::OPT_mno_warn_nonportable_cfstrings));
2671e5dd7070Spatrick       break;
2672e5dd7070Spatrick     }
2673e5dd7070Spatrick   }
2674e5dd7070Spatrick 
2675e5dd7070Spatrick   // Add the arch options based on the particular spelling of -arch, to match
2676*12c85518Srobert   // how the driver works.
2677e5dd7070Spatrick   if (!BoundArch.empty()) {
2678e5dd7070Spatrick     StringRef Name = BoundArch;
2679e5dd7070Spatrick     const Option MCpu = Opts.getOption(options::OPT_mcpu_EQ);
2680e5dd7070Spatrick     const Option MArch = Opts.getOption(clang::driver::options::OPT_march_EQ);
2681e5dd7070Spatrick 
2682e5dd7070Spatrick     // This code must be kept in sync with LLVM's getArchTypeForDarwinArch,
2683e5dd7070Spatrick     // which defines the list of which architectures we accept.
2684e5dd7070Spatrick     if (Name == "ppc")
2685e5dd7070Spatrick       ;
2686e5dd7070Spatrick     else if (Name == "ppc601")
2687e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "601");
2688e5dd7070Spatrick     else if (Name == "ppc603")
2689e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "603");
2690e5dd7070Spatrick     else if (Name == "ppc604")
2691e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "604");
2692e5dd7070Spatrick     else if (Name == "ppc604e")
2693e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "604e");
2694e5dd7070Spatrick     else if (Name == "ppc750")
2695e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "750");
2696e5dd7070Spatrick     else if (Name == "ppc7400")
2697e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "7400");
2698e5dd7070Spatrick     else if (Name == "ppc7450")
2699e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "7450");
2700e5dd7070Spatrick     else if (Name == "ppc970")
2701e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MCpu, "970");
2702e5dd7070Spatrick 
2703e5dd7070Spatrick     else if (Name == "ppc64" || Name == "ppc64le")
2704e5dd7070Spatrick       DAL->AddFlagArg(nullptr, Opts.getOption(options::OPT_m64));
2705e5dd7070Spatrick 
2706e5dd7070Spatrick     else if (Name == "i386")
2707e5dd7070Spatrick       ;
2708e5dd7070Spatrick     else if (Name == "i486")
2709e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "i486");
2710e5dd7070Spatrick     else if (Name == "i586")
2711e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "i586");
2712e5dd7070Spatrick     else if (Name == "i686")
2713e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "i686");
2714e5dd7070Spatrick     else if (Name == "pentium")
2715e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "pentium");
2716e5dd7070Spatrick     else if (Name == "pentium2")
2717e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "pentium2");
2718e5dd7070Spatrick     else if (Name == "pentpro")
2719e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "pentiumpro");
2720e5dd7070Spatrick     else if (Name == "pentIIm3")
2721e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "pentium2");
2722e5dd7070Spatrick 
2723e5dd7070Spatrick     else if (Name == "x86_64" || Name == "x86_64h")
2724e5dd7070Spatrick       DAL->AddFlagArg(nullptr, Opts.getOption(options::OPT_m64));
2725e5dd7070Spatrick 
2726e5dd7070Spatrick     else if (Name == "arm")
2727e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv4t");
2728e5dd7070Spatrick     else if (Name == "armv4t")
2729e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv4t");
2730e5dd7070Spatrick     else if (Name == "armv5")
2731e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv5tej");
2732e5dd7070Spatrick     else if (Name == "xscale")
2733e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "xscale");
2734e5dd7070Spatrick     else if (Name == "armv6")
2735e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv6k");
2736e5dd7070Spatrick     else if (Name == "armv6m")
2737e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv6m");
2738e5dd7070Spatrick     else if (Name == "armv7")
2739e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv7a");
2740e5dd7070Spatrick     else if (Name == "armv7em")
2741e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv7em");
2742e5dd7070Spatrick     else if (Name == "armv7k")
2743e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv7k");
2744e5dd7070Spatrick     else if (Name == "armv7m")
2745e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv7m");
2746e5dd7070Spatrick     else if (Name == "armv7s")
2747e5dd7070Spatrick       DAL->AddJoinedArg(nullptr, MArch, "armv7s");
2748e5dd7070Spatrick   }
2749e5dd7070Spatrick 
2750e5dd7070Spatrick   return DAL;
2751e5dd7070Spatrick }
2752e5dd7070Spatrick 
AddLinkRuntimeLibArgs(const ArgList & Args,ArgStringList & CmdArgs,bool ForceLinkBuiltinRT) const2753e5dd7070Spatrick void MachO::AddLinkRuntimeLibArgs(const ArgList &Args,
2754e5dd7070Spatrick                                   ArgStringList &CmdArgs,
2755e5dd7070Spatrick                                   bool ForceLinkBuiltinRT) const {
2756e5dd7070Spatrick   // Embedded targets are simple at the moment, not supporting sanitizers and
2757e5dd7070Spatrick   // with different libraries for each member of the product { static, PIC } x
2758e5dd7070Spatrick   // { hard-float, soft-float }
2759e5dd7070Spatrick   llvm::SmallString<32> CompilerRT = StringRef("");
2760e5dd7070Spatrick   CompilerRT +=
2761e5dd7070Spatrick       (tools::arm::getARMFloatABI(*this, Args) == tools::arm::FloatABI::Hard)
2762e5dd7070Spatrick           ? "hard"
2763e5dd7070Spatrick           : "soft";
2764e5dd7070Spatrick   CompilerRT += Args.hasArg(options::OPT_fPIC) ? "_pic" : "_static";
2765e5dd7070Spatrick 
2766e5dd7070Spatrick   AddLinkRuntimeLib(Args, CmdArgs, CompilerRT, RLO_IsEmbedded);
2767e5dd7070Spatrick }
2768e5dd7070Spatrick 
isAlignedAllocationUnavailable() const2769e5dd7070Spatrick bool Darwin::isAlignedAllocationUnavailable() const {
2770e5dd7070Spatrick   llvm::Triple::OSType OS;
2771e5dd7070Spatrick 
2772a9ac8606Spatrick   if (isTargetMacCatalyst())
2773a9ac8606Spatrick     return TargetVersion < alignedAllocMinVersion(llvm::Triple::MacOSX);
2774e5dd7070Spatrick   switch (TargetPlatform) {
2775e5dd7070Spatrick   case MacOS: // Earlier than 10.13.
2776e5dd7070Spatrick     OS = llvm::Triple::MacOSX;
2777e5dd7070Spatrick     break;
2778e5dd7070Spatrick   case IPhoneOS:
2779e5dd7070Spatrick     OS = llvm::Triple::IOS;
2780e5dd7070Spatrick     break;
2781e5dd7070Spatrick   case TvOS: // Earlier than 11.0.
2782e5dd7070Spatrick     OS = llvm::Triple::TvOS;
2783e5dd7070Spatrick     break;
2784e5dd7070Spatrick   case WatchOS: // Earlier than 4.0.
2785e5dd7070Spatrick     OS = llvm::Triple::WatchOS;
2786e5dd7070Spatrick     break;
2787*12c85518Srobert   case DriverKit: // Always available.
2788*12c85518Srobert     return false;
2789e5dd7070Spatrick   }
2790e5dd7070Spatrick 
2791e5dd7070Spatrick   return TargetVersion < alignedAllocMinVersion(OS);
2792e5dd7070Spatrick }
2793e5dd7070Spatrick 
addClangTargetOptions(const llvm::opt::ArgList & DriverArgs,llvm::opt::ArgStringList & CC1Args,Action::OffloadKind DeviceOffloadKind) const2794e5dd7070Spatrick void Darwin::addClangTargetOptions(const llvm::opt::ArgList &DriverArgs,
2795e5dd7070Spatrick                                    llvm::opt::ArgStringList &CC1Args,
2796e5dd7070Spatrick                                    Action::OffloadKind DeviceOffloadKind) const {
2797e5dd7070Spatrick   // Pass "-faligned-alloc-unavailable" only when the user hasn't manually
2798e5dd7070Spatrick   // enabled or disabled aligned allocations.
2799e5dd7070Spatrick   if (!DriverArgs.hasArgNoClaim(options::OPT_faligned_allocation,
2800e5dd7070Spatrick                                 options::OPT_fno_aligned_allocation) &&
2801e5dd7070Spatrick       isAlignedAllocationUnavailable())
2802e5dd7070Spatrick     CC1Args.push_back("-faligned-alloc-unavailable");
2803e5dd7070Spatrick 
2804*12c85518Srobert   addClangCC1ASTargetOptions(DriverArgs, CC1Args);
2805ec727ea7Spatrick 
2806ec727ea7Spatrick   // Enable compatibility mode for NSItemProviderCompletionHandler in
2807ec727ea7Spatrick   // Foundation/NSItemProvider.h.
2808ec727ea7Spatrick   CC1Args.push_back("-fcompatibility-qualified-id-block-type-checking");
2809a9ac8606Spatrick 
2810a9ac8606Spatrick   // Give static local variables in inline functions hidden visibility when
2811a9ac8606Spatrick   // -fvisibility-inlines-hidden is enabled.
2812a9ac8606Spatrick   if (!DriverArgs.getLastArgNoClaim(
2813a9ac8606Spatrick           options::OPT_fvisibility_inlines_hidden_static_local_var,
2814a9ac8606Spatrick           options::OPT_fno_visibility_inlines_hidden_static_local_var))
2815a9ac8606Spatrick     CC1Args.push_back("-fvisibility-inlines-hidden-static-local-var");
2816e5dd7070Spatrick }
2817e5dd7070Spatrick 
addClangCC1ASTargetOptions(const llvm::opt::ArgList & Args,llvm::opt::ArgStringList & CC1ASArgs) const2818*12c85518Srobert void Darwin::addClangCC1ASTargetOptions(
2819*12c85518Srobert     const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CC1ASArgs) const {
2820*12c85518Srobert   if (TargetVariantTriple) {
2821*12c85518Srobert     CC1ASArgs.push_back("-darwin-target-variant-triple");
2822*12c85518Srobert     CC1ASArgs.push_back(Args.MakeArgString(TargetVariantTriple->getTriple()));
2823*12c85518Srobert   }
2824*12c85518Srobert 
2825*12c85518Srobert   if (SDKInfo) {
2826*12c85518Srobert     /// Pass the SDK version to the compiler when the SDK information is
2827*12c85518Srobert     /// available.
2828*12c85518Srobert     auto EmitTargetSDKVersionArg = [&](const VersionTuple &V) {
2829*12c85518Srobert       std::string Arg;
2830*12c85518Srobert       llvm::raw_string_ostream OS(Arg);
2831*12c85518Srobert       OS << "-target-sdk-version=" << V;
2832*12c85518Srobert       CC1ASArgs.push_back(Args.MakeArgString(OS.str()));
2833*12c85518Srobert     };
2834*12c85518Srobert 
2835*12c85518Srobert     if (isTargetMacCatalyst()) {
2836*12c85518Srobert       if (const auto *MacOStoMacCatalystMapping = SDKInfo->getVersionMapping(
2837*12c85518Srobert               DarwinSDKInfo::OSEnvPair::macOStoMacCatalystPair())) {
2838*12c85518Srobert         std::optional<VersionTuple> SDKVersion = MacOStoMacCatalystMapping->map(
2839*12c85518Srobert             SDKInfo->getVersion(), minimumMacCatalystDeploymentTarget(),
2840*12c85518Srobert             std::nullopt);
2841*12c85518Srobert         EmitTargetSDKVersionArg(
2842*12c85518Srobert             SDKVersion ? *SDKVersion : minimumMacCatalystDeploymentTarget());
2843*12c85518Srobert       }
2844*12c85518Srobert     } else {
2845*12c85518Srobert       EmitTargetSDKVersionArg(SDKInfo->getVersion());
2846*12c85518Srobert     }
2847*12c85518Srobert 
2848*12c85518Srobert     /// Pass the target variant SDK version to the compiler when the SDK
2849*12c85518Srobert     /// information is available and is required for target variant.
2850*12c85518Srobert     if (TargetVariantTriple) {
2851*12c85518Srobert       if (isTargetMacCatalyst()) {
2852*12c85518Srobert         std::string Arg;
2853*12c85518Srobert         llvm::raw_string_ostream OS(Arg);
2854*12c85518Srobert         OS << "-darwin-target-variant-sdk-version=" << SDKInfo->getVersion();
2855*12c85518Srobert         CC1ASArgs.push_back(Args.MakeArgString(OS.str()));
2856*12c85518Srobert       } else if (const auto *MacOStoMacCatalystMapping =
2857*12c85518Srobert                      SDKInfo->getVersionMapping(
2858*12c85518Srobert                          DarwinSDKInfo::OSEnvPair::macOStoMacCatalystPair())) {
2859*12c85518Srobert         if (std::optional<VersionTuple> SDKVersion =
2860*12c85518Srobert                 MacOStoMacCatalystMapping->map(
2861*12c85518Srobert                     SDKInfo->getVersion(), minimumMacCatalystDeploymentTarget(),
2862*12c85518Srobert                     std::nullopt)) {
2863*12c85518Srobert           std::string Arg;
2864*12c85518Srobert           llvm::raw_string_ostream OS(Arg);
2865*12c85518Srobert           OS << "-darwin-target-variant-sdk-version=" << *SDKVersion;
2866*12c85518Srobert           CC1ASArgs.push_back(Args.MakeArgString(OS.str()));
2867*12c85518Srobert         }
2868*12c85518Srobert       }
2869*12c85518Srobert     }
2870*12c85518Srobert   }
2871*12c85518Srobert }
2872*12c85518Srobert 
2873e5dd7070Spatrick DerivedArgList *
TranslateArgs(const DerivedArgList & Args,StringRef BoundArch,Action::OffloadKind DeviceOffloadKind) const2874e5dd7070Spatrick Darwin::TranslateArgs(const DerivedArgList &Args, StringRef BoundArch,
2875e5dd7070Spatrick                       Action::OffloadKind DeviceOffloadKind) const {
2876e5dd7070Spatrick   // First get the generic Apple args, before moving onto Darwin-specific ones.
2877e5dd7070Spatrick   DerivedArgList *DAL =
2878e5dd7070Spatrick       MachO::TranslateArgs(Args, BoundArch, DeviceOffloadKind);
2879e5dd7070Spatrick 
2880e5dd7070Spatrick   // If no architecture is bound, none of the translations here are relevant.
2881e5dd7070Spatrick   if (BoundArch.empty())
2882e5dd7070Spatrick     return DAL;
2883e5dd7070Spatrick 
2884e5dd7070Spatrick   // Add an explicit version min argument for the deployment target. We do this
2885e5dd7070Spatrick   // after argument translation because -Xarch_ arguments may add a version min
2886e5dd7070Spatrick   // argument.
2887e5dd7070Spatrick   AddDeploymentTarget(*DAL);
2888e5dd7070Spatrick 
2889e5dd7070Spatrick   // For iOS 6, undo the translation to add -static for -mkernel/-fapple-kext.
2890e5dd7070Spatrick   // FIXME: It would be far better to avoid inserting those -static arguments,
2891e5dd7070Spatrick   // but we can't check the deployment target in the translation code until
2892e5dd7070Spatrick   // it is set here.
2893*12c85518Srobert   if (isTargetWatchOSBased() || isTargetDriverKit() ||
2894e5dd7070Spatrick       (isTargetIOSBased() && !isIPhoneOSVersionLT(6, 0))) {
2895e5dd7070Spatrick     for (ArgList::iterator it = DAL->begin(), ie = DAL->end(); it != ie; ) {
2896e5dd7070Spatrick       Arg *A = *it;
2897e5dd7070Spatrick       ++it;
2898e5dd7070Spatrick       if (A->getOption().getID() != options::OPT_mkernel &&
2899e5dd7070Spatrick           A->getOption().getID() != options::OPT_fapple_kext)
2900e5dd7070Spatrick         continue;
2901e5dd7070Spatrick       assert(it != ie && "unexpected argument translation");
2902e5dd7070Spatrick       A = *it;
2903e5dd7070Spatrick       assert(A->getOption().getID() == options::OPT_static &&
2904e5dd7070Spatrick              "missing expected -static argument");
2905e5dd7070Spatrick       *it = nullptr;
2906e5dd7070Spatrick       ++it;
2907e5dd7070Spatrick     }
2908e5dd7070Spatrick   }
2909e5dd7070Spatrick 
2910e5dd7070Spatrick   auto Arch = tools::darwin::getArchTypeForMachOArchName(BoundArch);
2911e5dd7070Spatrick   if ((Arch == llvm::Triple::arm || Arch == llvm::Triple::thumb)) {
2912e5dd7070Spatrick     if (Args.hasFlag(options::OPT_fomit_frame_pointer,
2913e5dd7070Spatrick                      options::OPT_fno_omit_frame_pointer, false))
2914e5dd7070Spatrick       getDriver().Diag(clang::diag::warn_drv_unsupported_opt_for_target)
2915e5dd7070Spatrick           << "-fomit-frame-pointer" << BoundArch;
2916e5dd7070Spatrick   }
2917e5dd7070Spatrick 
2918e5dd7070Spatrick   return DAL;
2919e5dd7070Spatrick }
2920e5dd7070Spatrick 
getDefaultUnwindTableLevel(const ArgList & Args) const2921*12c85518Srobert ToolChain::UnwindTableLevel MachO::getDefaultUnwindTableLevel(const ArgList &Args) const {
2922e5dd7070Spatrick   // Unwind tables are not emitted if -fno-exceptions is supplied (except when
2923e5dd7070Spatrick   // targeting x86_64).
2924*12c85518Srobert   if (getArch() == llvm::Triple::x86_64 ||
2925e5dd7070Spatrick       (GetExceptionModel(Args) != llvm::ExceptionHandling::SjLj &&
2926e5dd7070Spatrick        Args.hasFlag(options::OPT_fexceptions, options::OPT_fno_exceptions,
2927*12c85518Srobert                     true)))
2928*12c85518Srobert     return getArch() == llvm::Triple::aarch64 ? UnwindTableLevel::Synchronous
2929*12c85518Srobert                                               : UnwindTableLevel::Asynchronous;
2930*12c85518Srobert 
2931*12c85518Srobert   return UnwindTableLevel::None;
2932e5dd7070Spatrick }
2933e5dd7070Spatrick 
UseDwarfDebugFlags() const2934e5dd7070Spatrick bool MachO::UseDwarfDebugFlags() const {
2935e5dd7070Spatrick   if (const char *S = ::getenv("RC_DEBUG_OPTIONS"))
2936e5dd7070Spatrick     return S[0] != '\0';
2937e5dd7070Spatrick   return false;
2938e5dd7070Spatrick }
2939e5dd7070Spatrick 
GetGlobalDebugPathRemapping() const2940*12c85518Srobert std::string MachO::GetGlobalDebugPathRemapping() const {
2941*12c85518Srobert   if (const char *S = ::getenv("RC_DEBUG_PREFIX_MAP"))
2942*12c85518Srobert     return S;
2943*12c85518Srobert   return {};
2944*12c85518Srobert }
2945*12c85518Srobert 
GetExceptionModel(const ArgList & Args) const2946e5dd7070Spatrick llvm::ExceptionHandling Darwin::GetExceptionModel(const ArgList &Args) const {
2947e5dd7070Spatrick   // Darwin uses SjLj exceptions on ARM.
2948e5dd7070Spatrick   if (getTriple().getArch() != llvm::Triple::arm &&
2949e5dd7070Spatrick       getTriple().getArch() != llvm::Triple::thumb)
2950e5dd7070Spatrick     return llvm::ExceptionHandling::None;
2951e5dd7070Spatrick 
2952e5dd7070Spatrick   // Only watchOS uses the new DWARF/Compact unwinding method.
2953e5dd7070Spatrick   llvm::Triple Triple(ComputeLLVMTriple(Args));
2954e5dd7070Spatrick   if (Triple.isWatchABI())
2955e5dd7070Spatrick     return llvm::ExceptionHandling::DwarfCFI;
2956e5dd7070Spatrick 
2957e5dd7070Spatrick   return llvm::ExceptionHandling::SjLj;
2958e5dd7070Spatrick }
2959e5dd7070Spatrick 
SupportsEmbeddedBitcode() const2960e5dd7070Spatrick bool Darwin::SupportsEmbeddedBitcode() const {
2961e5dd7070Spatrick   assert(TargetInitialized && "Target not initialized!");
2962e5dd7070Spatrick   if (isTargetIPhoneOS() && isIPhoneOSVersionLT(6, 0))
2963e5dd7070Spatrick     return false;
2964e5dd7070Spatrick   return true;
2965e5dd7070Spatrick }
2966e5dd7070Spatrick 
isPICDefault() const2967e5dd7070Spatrick bool MachO::isPICDefault() const { return true; }
2968e5dd7070Spatrick 
isPIEDefault(const llvm::opt::ArgList & Args) const2969*12c85518Srobert bool MachO::isPIEDefault(const llvm::opt::ArgList &Args) const { return false; }
2970e5dd7070Spatrick 
isPICDefaultForced() const2971e5dd7070Spatrick bool MachO::isPICDefaultForced() const {
2972e5dd7070Spatrick   return (getArch() == llvm::Triple::x86_64 ||
2973e5dd7070Spatrick           getArch() == llvm::Triple::aarch64);
2974e5dd7070Spatrick }
2975e5dd7070Spatrick 
SupportsProfiling() const2976e5dd7070Spatrick bool MachO::SupportsProfiling() const {
2977e5dd7070Spatrick   // Profiling instrumentation is only supported on x86.
2978e5dd7070Spatrick   return getTriple().isX86();
2979e5dd7070Spatrick }
2980e5dd7070Spatrick 
addMinVersionArgs(const ArgList & Args,ArgStringList & CmdArgs) const2981e5dd7070Spatrick void Darwin::addMinVersionArgs(const ArgList &Args,
2982e5dd7070Spatrick                                ArgStringList &CmdArgs) const {
2983a9ac8606Spatrick   VersionTuple TargetVersion = getTripleTargetVersion();
2984e5dd7070Spatrick 
2985e5dd7070Spatrick   if (isTargetWatchOS())
2986e5dd7070Spatrick     CmdArgs.push_back("-watchos_version_min");
2987e5dd7070Spatrick   else if (isTargetWatchOSSimulator())
2988e5dd7070Spatrick     CmdArgs.push_back("-watchos_simulator_version_min");
2989e5dd7070Spatrick   else if (isTargetTvOS())
2990e5dd7070Spatrick     CmdArgs.push_back("-tvos_version_min");
2991e5dd7070Spatrick   else if (isTargetTvOSSimulator())
2992e5dd7070Spatrick     CmdArgs.push_back("-tvos_simulator_version_min");
2993*12c85518Srobert   else if (isTargetDriverKit())
2994*12c85518Srobert     CmdArgs.push_back("-driverkit_version_min");
2995e5dd7070Spatrick   else if (isTargetIOSSimulator())
2996e5dd7070Spatrick     CmdArgs.push_back("-ios_simulator_version_min");
2997e5dd7070Spatrick   else if (isTargetIOSBased())
2998e5dd7070Spatrick     CmdArgs.push_back("-iphoneos_version_min");
2999a9ac8606Spatrick   else if (isTargetMacCatalyst())
3000a9ac8606Spatrick     CmdArgs.push_back("-maccatalyst_version_min");
3001e5dd7070Spatrick   else {
3002e5dd7070Spatrick     assert(isTargetMacOS() && "unexpected target");
3003e5dd7070Spatrick     CmdArgs.push_back("-macosx_version_min");
3004e5dd7070Spatrick   }
3005e5dd7070Spatrick 
3006ec727ea7Spatrick   VersionTuple MinTgtVers = getEffectiveTriple().getMinimumSupportedOSVersion();
3007ec727ea7Spatrick   if (!MinTgtVers.empty() && MinTgtVers > TargetVersion)
3008ec727ea7Spatrick     TargetVersion = MinTgtVers;
3009e5dd7070Spatrick   CmdArgs.push_back(Args.MakeArgString(TargetVersion.getAsString()));
3010*12c85518Srobert   if (TargetVariantTriple) {
3011*12c85518Srobert     assert(isTargetMacOSBased() && "unexpected target");
3012*12c85518Srobert     VersionTuple VariantTargetVersion;
3013*12c85518Srobert     if (TargetVariantTriple->isMacOSX()) {
3014*12c85518Srobert       CmdArgs.push_back("-macosx_version_min");
3015*12c85518Srobert       TargetVariantTriple->getMacOSXVersion(VariantTargetVersion);
3016*12c85518Srobert     } else {
3017*12c85518Srobert       assert(TargetVariantTriple->isiOS() &&
3018*12c85518Srobert              TargetVariantTriple->isMacCatalystEnvironment() &&
3019*12c85518Srobert              "unexpected target variant triple");
3020*12c85518Srobert       CmdArgs.push_back("-maccatalyst_version_min");
3021*12c85518Srobert       VariantTargetVersion = TargetVariantTriple->getiOSVersion();
3022*12c85518Srobert     }
3023*12c85518Srobert     VersionTuple MinTgtVers =
3024*12c85518Srobert         TargetVariantTriple->getMinimumSupportedOSVersion();
3025*12c85518Srobert     if (MinTgtVers.getMajor() && MinTgtVers > VariantTargetVersion)
3026*12c85518Srobert       VariantTargetVersion = MinTgtVers;
3027*12c85518Srobert     CmdArgs.push_back(Args.MakeArgString(VariantTargetVersion.getAsString()));
3028*12c85518Srobert   }
3029e5dd7070Spatrick }
3030e5dd7070Spatrick 
getPlatformName(Darwin::DarwinPlatformKind Platform,Darwin::DarwinEnvironmentKind Environment)3031e5dd7070Spatrick static const char *getPlatformName(Darwin::DarwinPlatformKind Platform,
3032e5dd7070Spatrick                                    Darwin::DarwinEnvironmentKind Environment) {
3033e5dd7070Spatrick   switch (Platform) {
3034e5dd7070Spatrick   case Darwin::MacOS:
3035e5dd7070Spatrick     return "macos";
3036e5dd7070Spatrick   case Darwin::IPhoneOS:
3037a9ac8606Spatrick     if (Environment == Darwin::MacCatalyst)
3038a9ac8606Spatrick       return "mac catalyst";
3039e5dd7070Spatrick     return "ios";
3040e5dd7070Spatrick   case Darwin::TvOS:
3041e5dd7070Spatrick     return "tvos";
3042e5dd7070Spatrick   case Darwin::WatchOS:
3043e5dd7070Spatrick     return "watchos";
3044*12c85518Srobert   case Darwin::DriverKit:
3045*12c85518Srobert     return "driverkit";
3046e5dd7070Spatrick   }
3047e5dd7070Spatrick   llvm_unreachable("invalid platform");
3048e5dd7070Spatrick }
3049e5dd7070Spatrick 
addPlatformVersionArgs(const llvm::opt::ArgList & Args,llvm::opt::ArgStringList & CmdArgs) const3050e5dd7070Spatrick void Darwin::addPlatformVersionArgs(const llvm::opt::ArgList &Args,
3051e5dd7070Spatrick                                     llvm::opt::ArgStringList &CmdArgs) const {
3052*12c85518Srobert   auto EmitPlatformVersionArg =
3053*12c85518Srobert       [&](const VersionTuple &TV, Darwin::DarwinPlatformKind TargetPlatform,
3054*12c85518Srobert           Darwin::DarwinEnvironmentKind TargetEnvironment,
3055*12c85518Srobert           const llvm::Triple &TT) {
3056e5dd7070Spatrick         // -platform_version <platform> <target_version> <sdk_version>
3057e5dd7070Spatrick         // Both the target and SDK version support only up to 3 components.
3058e5dd7070Spatrick         CmdArgs.push_back("-platform_version");
3059*12c85518Srobert         std::string PlatformName =
3060*12c85518Srobert             getPlatformName(TargetPlatform, TargetEnvironment);
3061e5dd7070Spatrick         if (TargetEnvironment == Darwin::Simulator)
3062e5dd7070Spatrick           PlatformName += "-simulator";
3063e5dd7070Spatrick         CmdArgs.push_back(Args.MakeArgString(PlatformName));
3064*12c85518Srobert         VersionTuple TargetVersion = TV.withoutBuild();
3065*12c85518Srobert         if ((TargetPlatform == Darwin::IPhoneOS ||
3066*12c85518Srobert              TargetPlatform == Darwin::TvOS) &&
3067*12c85518Srobert             getTriple().getArchName() == "arm64e" &&
3068*12c85518Srobert             TargetVersion.getMajor() < 14) {
3069*12c85518Srobert           // arm64e slice is supported on iOS/tvOS 14+ only.
3070*12c85518Srobert           TargetVersion = VersionTuple(14, 0);
3071*12c85518Srobert         }
3072*12c85518Srobert         VersionTuple MinTgtVers = TT.getMinimumSupportedOSVersion();
3073ec727ea7Spatrick         if (!MinTgtVers.empty() && MinTgtVers > TargetVersion)
3074ec727ea7Spatrick           TargetVersion = MinTgtVers;
3075e5dd7070Spatrick         CmdArgs.push_back(Args.MakeArgString(TargetVersion.getAsString()));
3076a9ac8606Spatrick 
3077*12c85518Srobert         if (TargetPlatform == IPhoneOS && TargetEnvironment == MacCatalyst) {
3078a9ac8606Spatrick           // Mac Catalyst programs must use the appropriate iOS SDK version
3079a9ac8606Spatrick           // that corresponds to the macOS SDK version used for the compilation.
3080*12c85518Srobert           std::optional<VersionTuple> iOSSDKVersion;
3081a9ac8606Spatrick           if (SDKInfo) {
3082*12c85518Srobert             if (const auto *MacOStoMacCatalystMapping =
3083*12c85518Srobert                     SDKInfo->getVersionMapping(
3084a9ac8606Spatrick                         DarwinSDKInfo::OSEnvPair::macOStoMacCatalystPair())) {
3085a9ac8606Spatrick               iOSSDKVersion = MacOStoMacCatalystMapping->map(
3086a9ac8606Spatrick                   SDKInfo->getVersion().withoutBuild(),
3087*12c85518Srobert                   minimumMacCatalystDeploymentTarget(), std::nullopt);
3088a9ac8606Spatrick             }
3089a9ac8606Spatrick           }
3090a9ac8606Spatrick           CmdArgs.push_back(Args.MakeArgString(
3091*12c85518Srobert               (iOSSDKVersion ? *iOSSDKVersion
3092*12c85518Srobert                              : minimumMacCatalystDeploymentTarget())
3093a9ac8606Spatrick                   .getAsString()));
3094a9ac8606Spatrick           return;
3095a9ac8606Spatrick         }
3096a9ac8606Spatrick 
3097e5dd7070Spatrick         if (SDKInfo) {
3098e5dd7070Spatrick           VersionTuple SDKVersion = SDKInfo->getVersion().withoutBuild();
3099*12c85518Srobert           if (!SDKVersion.getMinor())
3100*12c85518Srobert             SDKVersion = VersionTuple(SDKVersion.getMajor(), 0);
3101e5dd7070Spatrick           CmdArgs.push_back(Args.MakeArgString(SDKVersion.getAsString()));
3102e5dd7070Spatrick         } else {
3103a9ac8606Spatrick           // Use an SDK version that's matching the deployment target if the SDK
3104*12c85518Srobert           // version is missing. This is preferred over an empty SDK version
3105*12c85518Srobert           // (0.0.0) as the system's runtime might expect the linked binary to
3106*12c85518Srobert           // contain a valid SDK version in order for the binary to work
3107*12c85518Srobert           // correctly. It's reasonable to use the deployment target version as
3108*12c85518Srobert           // a proxy for the SDK version because older SDKs don't guarantee
3109*12c85518Srobert           // support for deployment targets newer than the SDK versions, so that
3110*12c85518Srobert           // rules out using some predetermined older SDK version, which leaves
3111*12c85518Srobert           // the deployment target version as the only reasonable choice.
3112a9ac8606Spatrick           CmdArgs.push_back(Args.MakeArgString(TargetVersion.getAsString()));
3113e5dd7070Spatrick         }
3114*12c85518Srobert       };
3115*12c85518Srobert   EmitPlatformVersionArg(getTripleTargetVersion(), TargetPlatform,
3116*12c85518Srobert                          TargetEnvironment, getEffectiveTriple());
3117*12c85518Srobert   if (!TargetVariantTriple)
3118*12c85518Srobert     return;
3119*12c85518Srobert   Darwin::DarwinPlatformKind Platform;
3120*12c85518Srobert   Darwin::DarwinEnvironmentKind Environment;
3121*12c85518Srobert   VersionTuple TargetVariantVersion;
3122*12c85518Srobert   if (TargetVariantTriple->isMacOSX()) {
3123*12c85518Srobert     TargetVariantTriple->getMacOSXVersion(TargetVariantVersion);
3124*12c85518Srobert     Platform = Darwin::MacOS;
3125*12c85518Srobert     Environment = Darwin::NativeEnvironment;
3126*12c85518Srobert   } else {
3127*12c85518Srobert     assert(TargetVariantTriple->isiOS() &&
3128*12c85518Srobert            TargetVariantTriple->isMacCatalystEnvironment() &&
3129*12c85518Srobert            "unexpected target variant triple");
3130*12c85518Srobert     TargetVariantVersion = TargetVariantTriple->getiOSVersion();
3131*12c85518Srobert     Platform = Darwin::IPhoneOS;
3132*12c85518Srobert     Environment = Darwin::MacCatalyst;
3133*12c85518Srobert   }
3134*12c85518Srobert   EmitPlatformVersionArg(TargetVariantVersion, Platform, Environment,
3135*12c85518Srobert                          *TargetVariantTriple);
3136e5dd7070Spatrick }
3137e5dd7070Spatrick 
3138ec727ea7Spatrick // Add additional link args for the -dynamiclib option.
addDynamicLibLinkArgs(const Darwin & D,const ArgList & Args,ArgStringList & CmdArgs)3139ec727ea7Spatrick static void addDynamicLibLinkArgs(const Darwin &D, const ArgList &Args,
3140ec727ea7Spatrick                                   ArgStringList &CmdArgs) {
3141e5dd7070Spatrick   // Derived from darwin_dylib1 spec.
3142ec727ea7Spatrick   if (D.isTargetIPhoneOS()) {
3143ec727ea7Spatrick     if (D.isIPhoneOSVersionLT(3, 1))
3144e5dd7070Spatrick       CmdArgs.push_back("-ldylib1.o");
3145ec727ea7Spatrick     return;
3146ec727ea7Spatrick   }
3147ec727ea7Spatrick 
3148ec727ea7Spatrick   if (!D.isTargetMacOS())
3149ec727ea7Spatrick     return;
3150ec727ea7Spatrick   if (D.isMacosxVersionLT(10, 5))
3151e5dd7070Spatrick     CmdArgs.push_back("-ldylib1.o");
3152ec727ea7Spatrick   else if (D.isMacosxVersionLT(10, 6))
3153e5dd7070Spatrick     CmdArgs.push_back("-ldylib1.10.5.o");
3154e5dd7070Spatrick }
3155ec727ea7Spatrick 
3156ec727ea7Spatrick // Add additional link args for the -bundle option.
addBundleLinkArgs(const Darwin & D,const ArgList & Args,ArgStringList & CmdArgs)3157ec727ea7Spatrick static void addBundleLinkArgs(const Darwin &D, const ArgList &Args,
3158ec727ea7Spatrick                               ArgStringList &CmdArgs) {
3159ec727ea7Spatrick   if (Args.hasArg(options::OPT_static))
3160ec727ea7Spatrick     return;
3161e5dd7070Spatrick   // Derived from darwin_bundle1 spec.
3162ec727ea7Spatrick   if ((D.isTargetIPhoneOS() && D.isIPhoneOSVersionLT(3, 1)) ||
3163ec727ea7Spatrick       (D.isTargetMacOS() && D.isMacosxVersionLT(10, 6)))
3164e5dd7070Spatrick     CmdArgs.push_back("-lbundle1.o");
3165e5dd7070Spatrick }
3166ec727ea7Spatrick 
3167ec727ea7Spatrick // Add additional link args for the -pg option.
addPgProfilingLinkArgs(const Darwin & D,const ArgList & Args,ArgStringList & CmdArgs)3168ec727ea7Spatrick static void addPgProfilingLinkArgs(const Darwin &D, const ArgList &Args,
3169ec727ea7Spatrick                                    ArgStringList &CmdArgs) {
3170ec727ea7Spatrick   if (D.isTargetMacOS() && D.isMacosxVersionLT(10, 9)) {
3171ec727ea7Spatrick     if (Args.hasArg(options::OPT_static) || Args.hasArg(options::OPT_object) ||
3172e5dd7070Spatrick         Args.hasArg(options::OPT_preload)) {
3173e5dd7070Spatrick       CmdArgs.push_back("-lgcrt0.o");
3174e5dd7070Spatrick     } else {
3175e5dd7070Spatrick       CmdArgs.push_back("-lgcrt1.o");
3176e5dd7070Spatrick 
3177e5dd7070Spatrick       // darwin_crt2 spec is empty.
3178e5dd7070Spatrick     }
3179e5dd7070Spatrick     // By default on OS X 10.8 and later, we don't link with a crt1.o
3180e5dd7070Spatrick     // file and the linker knows to use _main as the entry point.  But,
3181e5dd7070Spatrick     // when compiling with -pg, we need to link with the gcrt1.o file,
3182e5dd7070Spatrick     // so pass the -no_new_main option to tell the linker to use the
3183e5dd7070Spatrick     // "start" symbol as the entry point.
3184ec727ea7Spatrick     if (!D.isMacosxVersionLT(10, 8))
3185e5dd7070Spatrick       CmdArgs.push_back("-no_new_main");
3186e5dd7070Spatrick   } else {
3187ec727ea7Spatrick     D.getDriver().Diag(diag::err_drv_clang_unsupported_opt_pg_darwin)
3188a9ac8606Spatrick         << D.isTargetMacOSBased();
3189e5dd7070Spatrick   }
3190ec727ea7Spatrick }
3191e5dd7070Spatrick 
addDefaultCRTLinkArgs(const Darwin & D,const ArgList & Args,ArgStringList & CmdArgs)3192ec727ea7Spatrick static void addDefaultCRTLinkArgs(const Darwin &D, const ArgList &Args,
3193ec727ea7Spatrick                                   ArgStringList &CmdArgs) {
3194ec727ea7Spatrick   // Derived from darwin_crt1 spec.
3195ec727ea7Spatrick   if (D.isTargetIPhoneOS()) {
3196ec727ea7Spatrick     if (D.getArch() == llvm::Triple::aarch64)
3197ec727ea7Spatrick       ; // iOS does not need any crt1 files for arm64
3198ec727ea7Spatrick     else if (D.isIPhoneOSVersionLT(3, 1))
3199ec727ea7Spatrick       CmdArgs.push_back("-lcrt1.o");
3200ec727ea7Spatrick     else if (D.isIPhoneOSVersionLT(6, 0))
3201ec727ea7Spatrick       CmdArgs.push_back("-lcrt1.3.1.o");
3202ec727ea7Spatrick     return;
3203ec727ea7Spatrick   }
3204ec727ea7Spatrick 
3205ec727ea7Spatrick   if (!D.isTargetMacOS())
3206ec727ea7Spatrick     return;
3207ec727ea7Spatrick   if (D.isMacosxVersionLT(10, 5))
3208ec727ea7Spatrick     CmdArgs.push_back("-lcrt1.o");
3209ec727ea7Spatrick   else if (D.isMacosxVersionLT(10, 6))
3210ec727ea7Spatrick     CmdArgs.push_back("-lcrt1.10.5.o");
3211ec727ea7Spatrick   else if (D.isMacosxVersionLT(10, 8))
3212ec727ea7Spatrick     CmdArgs.push_back("-lcrt1.10.6.o");
3213e5dd7070Spatrick   // darwin_crt2 spec is empty.
3214e5dd7070Spatrick }
3215e5dd7070Spatrick 
addStartObjectFileArgs(const ArgList & Args,ArgStringList & CmdArgs) const3216ec727ea7Spatrick void Darwin::addStartObjectFileArgs(const ArgList &Args,
3217ec727ea7Spatrick                                     ArgStringList &CmdArgs) const {
3218ec727ea7Spatrick   // Derived from startfile spec.
3219ec727ea7Spatrick   if (Args.hasArg(options::OPT_dynamiclib))
3220ec727ea7Spatrick     addDynamicLibLinkArgs(*this, Args, CmdArgs);
3221ec727ea7Spatrick   else if (Args.hasArg(options::OPT_bundle))
3222ec727ea7Spatrick     addBundleLinkArgs(*this, Args, CmdArgs);
3223ec727ea7Spatrick   else if (Args.hasArg(options::OPT_pg) && SupportsProfiling())
3224ec727ea7Spatrick     addPgProfilingLinkArgs(*this, Args, CmdArgs);
3225ec727ea7Spatrick   else if (Args.hasArg(options::OPT_static) ||
3226ec727ea7Spatrick            Args.hasArg(options::OPT_object) ||
3227ec727ea7Spatrick            Args.hasArg(options::OPT_preload))
3228ec727ea7Spatrick     CmdArgs.push_back("-lcrt0.o");
3229ec727ea7Spatrick   else
3230ec727ea7Spatrick     addDefaultCRTLinkArgs(*this, Args, CmdArgs);
3231ec727ea7Spatrick 
3232ec727ea7Spatrick   if (isTargetMacOS() && Args.hasArg(options::OPT_shared_libgcc) &&
3233ec727ea7Spatrick       isMacosxVersionLT(10, 5)) {
3234e5dd7070Spatrick     const char *Str = Args.MakeArgString(GetFilePath("crt3.o"));
3235e5dd7070Spatrick     CmdArgs.push_back(Str);
3236e5dd7070Spatrick   }
3237e5dd7070Spatrick }
3238e5dd7070Spatrick 
CheckObjCARC() const3239e5dd7070Spatrick void Darwin::CheckObjCARC() const {
3240e5dd7070Spatrick   if (isTargetIOSBased() || isTargetWatchOSBased() ||
3241a9ac8606Spatrick       (isTargetMacOSBased() && !isMacosxVersionLT(10, 6)))
3242e5dd7070Spatrick     return;
3243e5dd7070Spatrick   getDriver().Diag(diag::err_arc_unsupported_on_toolchain);
3244e5dd7070Spatrick }
3245e5dd7070Spatrick 
getSupportedSanitizers() const3246e5dd7070Spatrick SanitizerMask Darwin::getSupportedSanitizers() const {
3247e5dd7070Spatrick   const bool IsX86_64 = getTriple().getArch() == llvm::Triple::x86_64;
3248a9ac8606Spatrick   const bool IsAArch64 = getTriple().getArch() == llvm::Triple::aarch64;
3249e5dd7070Spatrick   SanitizerMask Res = ToolChain::getSupportedSanitizers();
3250e5dd7070Spatrick   Res |= SanitizerKind::Address;
3251e5dd7070Spatrick   Res |= SanitizerKind::PointerCompare;
3252e5dd7070Spatrick   Res |= SanitizerKind::PointerSubtract;
3253e5dd7070Spatrick   Res |= SanitizerKind::Leak;
3254e5dd7070Spatrick   Res |= SanitizerKind::Fuzzer;
3255e5dd7070Spatrick   Res |= SanitizerKind::FuzzerNoLink;
3256e5dd7070Spatrick   Res |= SanitizerKind::Function;
3257ec727ea7Spatrick   Res |= SanitizerKind::ObjCCast;
3258e5dd7070Spatrick 
3259e5dd7070Spatrick   // Prior to 10.9, macOS shipped a version of the C++ standard library without
3260e5dd7070Spatrick   // C++11 support. The same is true of iOS prior to version 5. These OS'es are
3261e5dd7070Spatrick   // incompatible with -fsanitize=vptr.
3262a9ac8606Spatrick   if (!(isTargetMacOSBased() && isMacosxVersionLT(10, 9)) &&
3263a9ac8606Spatrick       !(isTargetIPhoneOS() && isIPhoneOSVersionLT(5, 0)))
3264e5dd7070Spatrick     Res |= SanitizerKind::Vptr;
3265e5dd7070Spatrick 
3266*12c85518Srobert   if ((IsX86_64 || IsAArch64) &&
3267*12c85518Srobert       (isTargetMacOSBased() || isTargetIOSSimulator() ||
3268*12c85518Srobert        isTargetTvOSSimulator() || isTargetWatchOSSimulator())) {
3269e5dd7070Spatrick     Res |= SanitizerKind::Thread;
3270e5dd7070Spatrick   }
3271e5dd7070Spatrick   return Res;
3272e5dd7070Spatrick }
3273e5dd7070Spatrick 
printVerboseInfo(raw_ostream & OS) const3274e5dd7070Spatrick void Darwin::printVerboseInfo(raw_ostream &OS) const {
3275e5dd7070Spatrick   CudaInstallation.print(OS);
3276ec727ea7Spatrick   RocmInstallation.print(OS);
3277e5dd7070Spatrick }
3278