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