1 //===--- ARM.cpp - Implement ARM target feature support -------------------===//
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 // This file implements ARM TargetInfo objects.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "ARM.h"
14 #include "clang/Basic/Builtins.h"
15 #include "clang/Basic/Diagnostic.h"
16 #include "clang/Basic/TargetBuiltins.h"
17 #include "llvm/ADT/StringExtras.h"
18 #include "llvm/ADT/StringRef.h"
19 #include "llvm/ADT/StringSwitch.h"
20 
21 using namespace clang;
22 using namespace clang::targets;
23 
24 void ARMTargetInfo::setABIAAPCS() {
25   IsAAPCS = true;
26 
27   DoubleAlign = LongLongAlign = LongDoubleAlign = SuitableAlign = 64;
28   const llvm::Triple &T = getTriple();
29 
30   bool IsNetBSD = T.isOSNetBSD();
31   bool IsOpenBSD = T.isOSOpenBSD();
32   if (!T.isOSWindows() && !IsNetBSD && !IsOpenBSD)
33     WCharType = UnsignedInt;
34 
35   UseBitFieldTypeAlignment = true;
36 
37   ZeroLengthBitfieldBoundary = 0;
38 
39   // Thumb1 add sp, #imm requires the immediate value be multiple of 4,
40   // so set preferred for small types to 32.
41   if (T.isOSBinFormatMachO()) {
42     resetDataLayout(BigEndian
43                         ? "E-m:o-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
44                         : "e-m:o-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64");
45   } else if (T.isOSWindows()) {
46     assert(!BigEndian && "Windows on ARM does not support big endian");
47     resetDataLayout("e"
48                     "-m:w"
49                     "-p:32:32"
50                     "-Fi8"
51                     "-i64:64"
52                     "-v128:64:128"
53                     "-a:0:32"
54                     "-n32"
55                     "-S64");
56   } else if (T.isOSNaCl()) {
57     assert(!BigEndian && "NaCl on ARM does not support big endian");
58     resetDataLayout("e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S128");
59   } else {
60     resetDataLayout(BigEndian
61                         ? "E-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
62                         : "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64");
63   }
64 
65   // FIXME: Enumerated types are variable width in straight AAPCS.
66 }
67 
68 void ARMTargetInfo::setABIAPCS(bool IsAAPCS16) {
69   const llvm::Triple &T = getTriple();
70 
71   IsAAPCS = false;
72 
73   if (IsAAPCS16)
74     DoubleAlign = LongLongAlign = LongDoubleAlign = SuitableAlign = 64;
75   else
76     DoubleAlign = LongLongAlign = LongDoubleAlign = SuitableAlign = 32;
77 
78   WCharType = SignedInt;
79 
80   // Do not respect the alignment of bit-field types when laying out
81   // structures. This corresponds to PCC_BITFIELD_TYPE_MATTERS in gcc.
82   UseBitFieldTypeAlignment = false;
83 
84   /// gcc forces the alignment to 4 bytes, regardless of the type of the
85   /// zero length bitfield.  This corresponds to EMPTY_FIELD_BOUNDARY in
86   /// gcc.
87   ZeroLengthBitfieldBoundary = 32;
88 
89   if (T.isOSBinFormatMachO() && IsAAPCS16) {
90     assert(!BigEndian && "AAPCS16 does not support big-endian");
91     resetDataLayout("e-m:o-p:32:32-Fi8-i64:64-a:0:32-n32-S128");
92   } else if (T.isOSBinFormatMachO())
93     resetDataLayout(
94         BigEndian
95             ? "E-m:o-p:32:32-Fi8-f64:32:64-v64:32:64-v128:32:128-a:0:32-n32-S32"
96             : "e-m:o-p:32:32-Fi8-f64:32:64-v64:32:64-v128:32:128-a:0:32-n32-S32");
97   else
98     resetDataLayout(
99         BigEndian
100             ? "E-m:e-p:32:32-Fi8-f64:32:64-v64:32:64-v128:32:128-a:0:32-n32-S32"
101             : "e-m:e-p:32:32-Fi8-f64:32:64-v64:32:64-v128:32:128-a:0:32-n32-S32");
102 
103   // FIXME: Override "preferred align" for double and long long.
104 }
105 
106 void ARMTargetInfo::setArchInfo() {
107   StringRef ArchName = getTriple().getArchName();
108 
109   ArchISA = llvm::ARM::parseArchISA(ArchName);
110   CPU = llvm::ARM::getDefaultCPU(ArchName);
111   llvm::ARM::ArchKind AK = llvm::ARM::parseArch(ArchName);
112   if (AK != llvm::ARM::ArchKind::INVALID)
113     ArchKind = AK;
114   setArchInfo(ArchKind);
115 }
116 
117 void ARMTargetInfo::setArchInfo(llvm::ARM::ArchKind Kind) {
118   StringRef SubArch;
119 
120   // cache TargetParser info
121   ArchKind = Kind;
122   SubArch = llvm::ARM::getSubArch(ArchKind);
123   ArchProfile = llvm::ARM::parseArchProfile(SubArch);
124   ArchVersion = llvm::ARM::parseArchVersion(SubArch);
125 
126   // cache CPU related strings
127   CPUAttr = getCPUAttr();
128   CPUProfile = getCPUProfile();
129 }
130 
131 void ARMTargetInfo::setAtomic() {
132   // when triple does not specify a sub arch,
133   // then we are not using inline atomics
134   bool ShouldUseInlineAtomic =
135       (ArchISA == llvm::ARM::ISAKind::ARM && ArchVersion >= 6) ||
136       (ArchISA == llvm::ARM::ISAKind::THUMB && ArchVersion >= 7);
137   // Cortex M does not support 8 byte atomics, while general Thumb2 does.
138   if (ArchProfile == llvm::ARM::ProfileKind::M) {
139     MaxAtomicPromoteWidth = 32;
140     if (ShouldUseInlineAtomic)
141       MaxAtomicInlineWidth = 32;
142   } else {
143     MaxAtomicPromoteWidth = 64;
144     if (ShouldUseInlineAtomic)
145       MaxAtomicInlineWidth = 64;
146   }
147 }
148 
149 bool ARMTargetInfo::hasMVE() const {
150   return ArchKind == llvm::ARM::ArchKind::ARMV8_1MMainline && MVE != 0;
151 }
152 
153 bool ARMTargetInfo::hasMVEFloat() const {
154   return hasMVE() && (MVE & MVE_FP);
155 }
156 
157 bool ARMTargetInfo::isThumb() const {
158   return ArchISA == llvm::ARM::ISAKind::THUMB;
159 }
160 
161 bool ARMTargetInfo::supportsThumb() const {
162   return CPUAttr.count('T') || ArchVersion >= 6;
163 }
164 
165 bool ARMTargetInfo::supportsThumb2() const {
166   return CPUAttr.equals("6T2") ||
167          (ArchVersion >= 7 && !CPUAttr.equals("8M_BASE"));
168 }
169 
170 StringRef ARMTargetInfo::getCPUAttr() const {
171   // For most sub-arches, the build attribute CPU name is enough.
172   // For Cortex variants, it's slightly different.
173   switch (ArchKind) {
174   default:
175     return llvm::ARM::getCPUAttr(ArchKind);
176   case llvm::ARM::ArchKind::ARMV6M:
177     return "6M";
178   case llvm::ARM::ArchKind::ARMV7S:
179     return "7S";
180   case llvm::ARM::ArchKind::ARMV7A:
181     return "7A";
182   case llvm::ARM::ArchKind::ARMV7R:
183     return "7R";
184   case llvm::ARM::ArchKind::ARMV7M:
185     return "7M";
186   case llvm::ARM::ArchKind::ARMV7EM:
187     return "7EM";
188   case llvm::ARM::ArchKind::ARMV7VE:
189     return "7VE";
190   case llvm::ARM::ArchKind::ARMV8A:
191     return "8A";
192   case llvm::ARM::ArchKind::ARMV8_1A:
193     return "8_1A";
194   case llvm::ARM::ArchKind::ARMV8_2A:
195     return "8_2A";
196   case llvm::ARM::ArchKind::ARMV8_3A:
197     return "8_3A";
198   case llvm::ARM::ArchKind::ARMV8_4A:
199     return "8_4A";
200   case llvm::ARM::ArchKind::ARMV8_5A:
201     return "8_5A";
202   case llvm::ARM::ArchKind::ARMV8MBaseline:
203     return "8M_BASE";
204   case llvm::ARM::ArchKind::ARMV8MMainline:
205     return "8M_MAIN";
206   case llvm::ARM::ArchKind::ARMV8R:
207     return "8R";
208   case llvm::ARM::ArchKind::ARMV8_1MMainline:
209     return "8_1M_MAIN";
210   }
211 }
212 
213 StringRef ARMTargetInfo::getCPUProfile() const {
214   switch (ArchProfile) {
215   case llvm::ARM::ProfileKind::A:
216     return "A";
217   case llvm::ARM::ProfileKind::R:
218     return "R";
219   case llvm::ARM::ProfileKind::M:
220     return "M";
221   default:
222     return "";
223   }
224 }
225 
226 ARMTargetInfo::ARMTargetInfo(const llvm::Triple &Triple,
227                              const TargetOptions &Opts)
228     : TargetInfo(Triple), FPMath(FP_Default), IsAAPCS(true), LDREX(0),
229       HW_FP(0) {
230   bool IsOpenBSD = Triple.isOSOpenBSD();
231   bool IsNetBSD = Triple.isOSNetBSD();
232 
233   // FIXME: the isOSBinFormatMachO is a workaround for identifying a Darwin-like
234   // environment where size_t is `unsigned long` rather than `unsigned int`
235 
236   PtrDiffType = IntPtrType =
237       (Triple.isOSDarwin() || Triple.isOSBinFormatMachO() || IsOpenBSD ||
238        IsNetBSD)
239           ? SignedLong
240           : SignedInt;
241 
242   SizeType = (Triple.isOSDarwin() || Triple.isOSBinFormatMachO() || IsOpenBSD ||
243               IsNetBSD)
244                  ? UnsignedLong
245                  : UnsignedInt;
246 
247   // ptrdiff_t is inconsistent on Darwin
248   if ((Triple.isOSDarwin() || Triple.isOSBinFormatMachO()) &&
249       !Triple.isWatchABI())
250     PtrDiffType = SignedInt;
251 
252   // Cache arch related info.
253   setArchInfo();
254 
255   // {} in inline assembly are neon specifiers, not assembly variant
256   // specifiers.
257   NoAsmVariants = true;
258 
259   // FIXME: This duplicates code from the driver that sets the -target-abi
260   // option - this code is used if -target-abi isn't passed and should
261   // be unified in some way.
262   if (Triple.isOSBinFormatMachO()) {
263     // The backend is hardwired to assume AAPCS for M-class processors, ensure
264     // the frontend matches that.
265     if (Triple.getEnvironment() == llvm::Triple::EABI ||
266         Triple.getOS() == llvm::Triple::UnknownOS ||
267         ArchProfile == llvm::ARM::ProfileKind::M) {
268       setABI("aapcs");
269     } else if (Triple.isWatchABI()) {
270       setABI("aapcs16");
271     } else {
272       setABI("apcs-gnu");
273     }
274   } else if (Triple.isOSWindows()) {
275     // FIXME: this is invalid for WindowsCE
276     setABI("aapcs");
277   } else {
278     // Select the default based on the platform.
279     switch (Triple.getEnvironment()) {
280     case llvm::Triple::Android:
281     case llvm::Triple::GNUEABI:
282     case llvm::Triple::GNUEABIHF:
283     case llvm::Triple::MuslEABI:
284     case llvm::Triple::MuslEABIHF:
285       setABI("aapcs-linux");
286       break;
287     case llvm::Triple::EABIHF:
288     case llvm::Triple::EABI:
289       setABI("aapcs");
290       break;
291     case llvm::Triple::GNU:
292       setABI("apcs-gnu");
293       break;
294     default:
295       if (IsNetBSD)
296         setABI("apcs-gnu");
297       else if (IsOpenBSD)
298         setABI("aapcs-linux");
299       else
300         setABI("aapcs");
301       break;
302     }
303   }
304 
305   // ARM targets default to using the ARM C++ ABI.
306   TheCXXABI.set(TargetCXXABI::GenericARM);
307 
308   // ARM has atomics up to 8 bytes
309   setAtomic();
310 
311   // Maximum alignment for ARM NEON data types should be 64-bits (AAPCS)
312   // as well the default alignment
313   if (IsAAPCS && (Triple.getEnvironment() != llvm::Triple::Android))
314     DefaultAlignForAttributeAligned = MaxVectorAlign = 64;
315 
316   // Do force alignment of members that follow zero length bitfields.  If
317   // the alignment of the zero-length bitfield is greater than the member
318   // that follows it, `bar', `bar' will be aligned as the  type of the
319   // zero length bitfield.
320   UseZeroLengthBitfieldAlignment = true;
321 
322   if (Triple.getOS() == llvm::Triple::Linux ||
323       Triple.getOS() == llvm::Triple::UnknownOS)
324     this->MCountName = Opts.EABIVersion == llvm::EABI::GNU
325                            ? "llvm.arm.gnu.eabi.mcount"
326                            : "\01mcount";
327 
328   SoftFloatABI = llvm::is_contained(Opts.FeaturesAsWritten, "+soft-float-abi");
329 }
330 
331 StringRef ARMTargetInfo::getABI() const { return ABI; }
332 
333 bool ARMTargetInfo::setABI(const std::string &Name) {
334   ABI = Name;
335 
336   // The defaults (above) are for AAPCS, check if we need to change them.
337   //
338   // FIXME: We need support for -meabi... we could just mangle it into the
339   // name.
340   if (Name == "apcs-gnu" || Name == "aapcs16") {
341     setABIAPCS(Name == "aapcs16");
342     return true;
343   }
344   if (Name == "aapcs" || Name == "aapcs-vfp" || Name == "aapcs-linux") {
345     setABIAAPCS();
346     return true;
347   }
348   return false;
349 }
350 
351 // FIXME: This should be based on Arch attributes, not CPU names.
352 bool ARMTargetInfo::initFeatureMap(
353     llvm::StringMap<bool> &Features, DiagnosticsEngine &Diags, StringRef CPU,
354     const std::vector<std::string> &FeaturesVec) const {
355 
356   std::string ArchFeature;
357   std::vector<StringRef> TargetFeatures;
358   llvm::ARM::ArchKind Arch = llvm::ARM::parseArch(getTriple().getArchName());
359 
360   // Map the base architecture to an appropriate target feature, so we don't
361   // rely on the target triple.
362   llvm::ARM::ArchKind CPUArch = llvm::ARM::parseCPUArch(CPU);
363   if (CPUArch == llvm::ARM::ArchKind::INVALID)
364     CPUArch = Arch;
365   if (CPUArch != llvm::ARM::ArchKind::INVALID) {
366     ArchFeature = ("+" + llvm::ARM::getArchName(CPUArch)).str();
367     TargetFeatures.push_back(ArchFeature);
368   }
369 
370   // get default FPU features
371   unsigned FPUKind = llvm::ARM::getDefaultFPU(CPU, Arch);
372   llvm::ARM::getFPUFeatures(FPUKind, TargetFeatures);
373 
374   // get default Extension features
375   unsigned Extensions = llvm::ARM::getDefaultExtensions(CPU, Arch);
376   llvm::ARM::getExtensionFeatures(Extensions, TargetFeatures);
377 
378   for (auto Feature : TargetFeatures)
379     if (Feature[0] == '+')
380       Features[Feature.drop_front(1)] = true;
381 
382   // Enable or disable thumb-mode explicitly per function to enable mixed
383   // ARM and Thumb code generation.
384   if (isThumb())
385     Features["thumb-mode"] = true;
386   else
387     Features["thumb-mode"] = false;
388 
389   // Convert user-provided arm and thumb GNU target attributes to
390   // [-|+]thumb-mode target features respectively.
391   std::vector<std::string> UpdatedFeaturesVec;
392   for (const auto &Feature : FeaturesVec) {
393     // Skip soft-float-abi; it's something we only use to initialize a bit of
394     // class state, and is otherwise unrecognized.
395     if (Feature == "+soft-float-abi")
396       continue;
397 
398     StringRef FixedFeature;
399     if (Feature == "+arm")
400       FixedFeature = "-thumb-mode";
401     else if (Feature == "+thumb")
402       FixedFeature = "+thumb-mode";
403     else
404       FixedFeature = Feature;
405     UpdatedFeaturesVec.push_back(FixedFeature.str());
406   }
407 
408   return TargetInfo::initFeatureMap(Features, Diags, CPU, UpdatedFeaturesVec);
409 }
410 
411 
412 bool ARMTargetInfo::handleTargetFeatures(std::vector<std::string> &Features,
413                                          DiagnosticsEngine &Diags) {
414   FPU = 0;
415   MVE = 0;
416   CRC = 0;
417   Crypto = 0;
418   DSP = 0;
419   Unaligned = 1;
420   SoftFloat = false;
421   // Note that SoftFloatABI is initialized in our constructor.
422   HWDiv = 0;
423   DotProd = 0;
424   HasFloat16 = true;
425 
426   // This does not diagnose illegal cases like having both
427   // "+vfpv2" and "+vfpv3" or having "+neon" and "-fp64".
428   for (const auto &Feature : Features) {
429     if (Feature == "+soft-float") {
430       SoftFloat = true;
431     } else if (Feature == "+vfp2sp" || Feature == "+vfp2") {
432       FPU |= VFP2FPU;
433       HW_FP |= HW_FP_SP;
434       if (Feature == "+vfp2")
435           HW_FP |= HW_FP_DP;
436     } else if (Feature == "+vfp3sp" || Feature == "+vfp3d16sp" ||
437                Feature == "+vfp3" || Feature == "+vfp3d16") {
438       FPU |= VFP3FPU;
439       HW_FP |= HW_FP_SP;
440       if (Feature == "+vfp3" || Feature == "+vfp3d16")
441           HW_FP |= HW_FP_DP;
442     } else if (Feature == "+vfp4sp" || Feature == "+vfp4d16sp" ||
443                Feature == "+vfp4" || Feature == "+vfp4d16") {
444       FPU |= VFP4FPU;
445       HW_FP |= HW_FP_SP | HW_FP_HP;
446       if (Feature == "+vfp4" || Feature == "+vfp4d16")
447           HW_FP |= HW_FP_DP;
448     } else if (Feature == "+fp-armv8sp" || Feature == "+fp-armv8d16sp" ||
449                Feature == "+fp-armv8" || Feature == "+fp-armv8d16") {
450       FPU |= FPARMV8;
451       HW_FP |= HW_FP_SP | HW_FP_HP;
452       if (Feature == "+fp-armv8" || Feature == "+fp-armv8d16")
453           HW_FP |= HW_FP_DP;
454     } else if (Feature == "+neon") {
455       FPU |= NeonFPU;
456       HW_FP |= HW_FP_SP;
457     } else if (Feature == "+hwdiv") {
458       HWDiv |= HWDivThumb;
459     } else if (Feature == "+hwdiv-arm") {
460       HWDiv |= HWDivARM;
461     } else if (Feature == "+crc") {
462       CRC = 1;
463     } else if (Feature == "+crypto") {
464       Crypto = 1;
465     } else if (Feature == "+dsp") {
466       DSP = 1;
467     } else if (Feature == "+fp64") {
468       HW_FP |= HW_FP_DP;
469     } else if (Feature == "+8msecext") {
470       if (CPUProfile != "M" || ArchVersion != 8) {
471         Diags.Report(diag::err_target_unsupported_mcmse) << CPU;
472         return false;
473       }
474     } else if (Feature == "+strict-align") {
475       Unaligned = 0;
476     } else if (Feature == "+fp16") {
477       HW_FP |= HW_FP_HP;
478     } else if (Feature == "+fullfp16") {
479       HasLegalHalfType = true;
480     } else if (Feature == "+dotprod") {
481       DotProd = true;
482     } else if (Feature == "+mve") {
483       DSP = 1;
484       MVE |= MVE_INT;
485     } else if (Feature == "+mve.fp") {
486       DSP = 1;
487       HasLegalHalfType = true;
488       FPU |= FPARMV8;
489       MVE |= MVE_INT | MVE_FP;
490       HW_FP |= HW_FP_SP | HW_FP_HP;
491     }
492   }
493 
494   switch (ArchVersion) {
495   case 6:
496     if (ArchProfile == llvm::ARM::ProfileKind::M)
497       LDREX = 0;
498     else if (ArchKind == llvm::ARM::ArchKind::ARMV6K)
499       LDREX = LDREX_D | LDREX_W | LDREX_H | LDREX_B;
500     else
501       LDREX = LDREX_W;
502     break;
503   case 7:
504     if (ArchProfile == llvm::ARM::ProfileKind::M)
505       LDREX = LDREX_W | LDREX_H | LDREX_B;
506     else
507       LDREX = LDREX_D | LDREX_W | LDREX_H | LDREX_B;
508     break;
509   case 8:
510     LDREX = LDREX_D | LDREX_W | LDREX_H | LDREX_B;
511   }
512 
513   if (!(FPU & NeonFPU) && FPMath == FP_Neon) {
514     Diags.Report(diag::err_target_unsupported_fpmath) << "neon";
515     return false;
516   }
517 
518   if (FPMath == FP_Neon)
519     Features.push_back("+neonfp");
520   else if (FPMath == FP_VFP)
521     Features.push_back("-neonfp");
522 
523   return true;
524 }
525 
526 bool ARMTargetInfo::hasFeature(StringRef Feature) const {
527   return llvm::StringSwitch<bool>(Feature)
528       .Case("arm", true)
529       .Case("aarch32", true)
530       .Case("softfloat", SoftFloat)
531       .Case("thumb", isThumb())
532       .Case("neon", (FPU & NeonFPU) && !SoftFloat)
533       .Case("vfp", FPU && !SoftFloat)
534       .Case("hwdiv", HWDiv & HWDivThumb)
535       .Case("hwdiv-arm", HWDiv & HWDivARM)
536       .Case("mve", hasMVE())
537       .Default(false);
538 }
539 
540 bool ARMTargetInfo::isValidCPUName(StringRef Name) const {
541   return Name == "generic" ||
542          llvm::ARM::parseCPUArch(Name) != llvm::ARM::ArchKind::INVALID;
543 }
544 
545 void ARMTargetInfo::fillValidCPUList(SmallVectorImpl<StringRef> &Values) const {
546   llvm::ARM::fillValidCPUArchList(Values);
547 }
548 
549 bool ARMTargetInfo::setCPU(const std::string &Name) {
550   if (Name != "generic")
551     setArchInfo(llvm::ARM::parseCPUArch(Name));
552 
553   if (ArchKind == llvm::ARM::ArchKind::INVALID)
554     return false;
555   setAtomic();
556   CPU = Name;
557   return true;
558 }
559 
560 bool ARMTargetInfo::setFPMath(StringRef Name) {
561   if (Name == "neon") {
562     FPMath = FP_Neon;
563     return true;
564   } else if (Name == "vfp" || Name == "vfp2" || Name == "vfp3" ||
565              Name == "vfp4") {
566     FPMath = FP_VFP;
567     return true;
568   }
569   return false;
570 }
571 
572 void ARMTargetInfo::getTargetDefinesARMV81A(const LangOptions &Opts,
573                                             MacroBuilder &Builder) const {
574   Builder.defineMacro("__ARM_FEATURE_QRDMX", "1");
575 }
576 
577 void ARMTargetInfo::getTargetDefinesARMV82A(const LangOptions &Opts,
578                                             MacroBuilder &Builder) const {
579   // Also include the ARMv8.1-A defines
580   getTargetDefinesARMV81A(Opts, Builder);
581 }
582 
583 void ARMTargetInfo::getTargetDefinesARMV83A(const LangOptions &Opts,
584                                             MacroBuilder &Builder) const {
585   // Also include the ARMv8.2-A defines
586   Builder.defineMacro("__ARM_FEATURE_COMPLEX", "1");
587   getTargetDefinesARMV82A(Opts, Builder);
588 }
589 
590 void ARMTargetInfo::getTargetDefines(const LangOptions &Opts,
591                                      MacroBuilder &Builder) const {
592   // Target identification.
593   Builder.defineMacro("__arm");
594   Builder.defineMacro("__arm__");
595   // For bare-metal none-eabi.
596   if (getTriple().getOS() == llvm::Triple::UnknownOS &&
597       (getTriple().getEnvironment() == llvm::Triple::EABI ||
598        getTriple().getEnvironment() == llvm::Triple::EABIHF))
599     Builder.defineMacro("__ELF__");
600 
601   // Target properties.
602   Builder.defineMacro("__REGISTER_PREFIX__", "");
603 
604   // Unfortunately, __ARM_ARCH_7K__ is now more of an ABI descriptor. The CPU
605   // happens to be Cortex-A7 though, so it should still get __ARM_ARCH_7A__.
606   if (getTriple().isWatchABI())
607     Builder.defineMacro("__ARM_ARCH_7K__", "2");
608 
609   if (!CPUAttr.empty())
610     Builder.defineMacro("__ARM_ARCH_" + CPUAttr + "__");
611 
612   // ACLE 6.4.1 ARM/Thumb instruction set architecture
613   // __ARM_ARCH is defined as an integer value indicating the current ARM ISA
614   Builder.defineMacro("__ARM_ARCH", Twine(ArchVersion));
615 
616   if (ArchVersion >= 8) {
617     // ACLE 6.5.7 Crypto Extension
618     if (Crypto)
619       Builder.defineMacro("__ARM_FEATURE_CRYPTO", "1");
620     // ACLE 6.5.8 CRC32 Extension
621     if (CRC)
622       Builder.defineMacro("__ARM_FEATURE_CRC32", "1");
623     // ACLE 6.5.10 Numeric Maximum and Minimum
624     Builder.defineMacro("__ARM_FEATURE_NUMERIC_MAXMIN", "1");
625     // ACLE 6.5.9 Directed Rounding
626     Builder.defineMacro("__ARM_FEATURE_DIRECTED_ROUNDING", "1");
627   }
628 
629   // __ARM_ARCH_ISA_ARM is defined to 1 if the core supports the ARM ISA.  It
630   // is not defined for the M-profile.
631   // NOTE that the default profile is assumed to be 'A'
632   if (CPUProfile.empty() || ArchProfile != llvm::ARM::ProfileKind::M)
633     Builder.defineMacro("__ARM_ARCH_ISA_ARM", "1");
634 
635   // __ARM_ARCH_ISA_THUMB is defined to 1 if the core supports the original
636   // Thumb ISA (including v6-M and v8-M Baseline).  It is set to 2 if the
637   // core supports the Thumb-2 ISA as found in the v6T2 architecture and all
638   // v7 and v8 architectures excluding v8-M Baseline.
639   if (supportsThumb2())
640     Builder.defineMacro("__ARM_ARCH_ISA_THUMB", "2");
641   else if (supportsThumb())
642     Builder.defineMacro("__ARM_ARCH_ISA_THUMB", "1");
643 
644   // __ARM_32BIT_STATE is defined to 1 if code is being generated for a 32-bit
645   // instruction set such as ARM or Thumb.
646   Builder.defineMacro("__ARM_32BIT_STATE", "1");
647 
648   // ACLE 6.4.2 Architectural Profile (A, R, M or pre-Cortex)
649 
650   // __ARM_ARCH_PROFILE is defined as 'A', 'R', 'M' or 'S', or unset.
651   if (!CPUProfile.empty())
652     Builder.defineMacro("__ARM_ARCH_PROFILE", "'" + CPUProfile + "'");
653 
654   // ACLE 6.4.3 Unaligned access supported in hardware
655   if (Unaligned)
656     Builder.defineMacro("__ARM_FEATURE_UNALIGNED", "1");
657 
658   // ACLE 6.4.4 LDREX/STREX
659   if (LDREX)
660     Builder.defineMacro("__ARM_FEATURE_LDREX", "0x" + Twine::utohexstr(LDREX));
661 
662   // ACLE 6.4.5 CLZ
663   if (ArchVersion == 5 || (ArchVersion == 6 && CPUProfile != "M") ||
664       ArchVersion > 6)
665     Builder.defineMacro("__ARM_FEATURE_CLZ", "1");
666 
667   // ACLE 6.5.1 Hardware Floating Point
668   if (HW_FP)
669     Builder.defineMacro("__ARM_FP", "0x" + Twine::utohexstr(HW_FP));
670 
671   // ACLE predefines.
672   Builder.defineMacro("__ARM_ACLE", "200");
673 
674   // FP16 support (we currently only support IEEE format).
675   Builder.defineMacro("__ARM_FP16_FORMAT_IEEE", "1");
676   Builder.defineMacro("__ARM_FP16_ARGS", "1");
677 
678   // ACLE 6.5.3 Fused multiply-accumulate (FMA)
679   if (ArchVersion >= 7 && (FPU & VFP4FPU))
680     Builder.defineMacro("__ARM_FEATURE_FMA", "1");
681 
682   // Subtarget options.
683 
684   // FIXME: It's more complicated than this and we don't really support
685   // interworking.
686   // Windows on ARM does not "support" interworking
687   if (5 <= ArchVersion && ArchVersion <= 8 && !getTriple().isOSWindows())
688     Builder.defineMacro("__THUMB_INTERWORK__");
689 
690   if (ABI == "aapcs" || ABI == "aapcs-linux" || ABI == "aapcs-vfp") {
691     // Embedded targets on Darwin follow AAPCS, but not EABI.
692     // Windows on ARM follows AAPCS VFP, but does not conform to EABI.
693     if (!getTriple().isOSBinFormatMachO() && !getTriple().isOSWindows())
694       Builder.defineMacro("__ARM_EABI__");
695     Builder.defineMacro("__ARM_PCS", "1");
696   }
697 
698   if ((!SoftFloat && !SoftFloatABI) || ABI == "aapcs-vfp" || ABI == "aapcs16")
699     Builder.defineMacro("__ARM_PCS_VFP", "1");
700 
701   if (SoftFloat)
702     Builder.defineMacro("__SOFTFP__");
703 
704   // ACLE position independent code macros.
705   if (Opts.ROPI)
706     Builder.defineMacro("__ARM_ROPI", "1");
707   if (Opts.RWPI)
708     Builder.defineMacro("__ARM_RWPI", "1");
709 
710   if (ArchKind == llvm::ARM::ArchKind::XSCALE)
711     Builder.defineMacro("__XSCALE__");
712 
713   if (isThumb()) {
714     Builder.defineMacro("__THUMBEL__");
715     Builder.defineMacro("__thumb__");
716     if (supportsThumb2())
717       Builder.defineMacro("__thumb2__");
718   }
719 
720   // ACLE 6.4.9 32-bit SIMD instructions
721   if ((CPUProfile != "M" && ArchVersion >= 6) || (CPUProfile == "M" && DSP))
722     Builder.defineMacro("__ARM_FEATURE_SIMD32", "1");
723 
724   // ACLE 6.4.10 Hardware Integer Divide
725   if (((HWDiv & HWDivThumb) && isThumb()) ||
726       ((HWDiv & HWDivARM) && !isThumb())) {
727     Builder.defineMacro("__ARM_FEATURE_IDIV", "1");
728     Builder.defineMacro("__ARM_ARCH_EXT_IDIV__", "1");
729   }
730 
731   // Note, this is always on in gcc, even though it doesn't make sense.
732   Builder.defineMacro("__APCS_32__");
733 
734   if (FPUModeIsVFP((FPUMode)FPU)) {
735     Builder.defineMacro("__VFP_FP__");
736     if (FPU & VFP2FPU)
737       Builder.defineMacro("__ARM_VFPV2__");
738     if (FPU & VFP3FPU)
739       Builder.defineMacro("__ARM_VFPV3__");
740     if (FPU & VFP4FPU)
741       Builder.defineMacro("__ARM_VFPV4__");
742     if (FPU & FPARMV8)
743       Builder.defineMacro("__ARM_FPV5__");
744   }
745 
746   // This only gets set when Neon instructions are actually available, unlike
747   // the VFP define, hence the soft float and arch check. This is subtly
748   // different from gcc, we follow the intent which was that it should be set
749   // when Neon instructions are actually available.
750   if ((FPU & NeonFPU) && !SoftFloat && ArchVersion >= 7) {
751     Builder.defineMacro("__ARM_NEON", "1");
752     Builder.defineMacro("__ARM_NEON__");
753     // current AArch32 NEON implementations do not support double-precision
754     // floating-point even when it is present in VFP.
755     Builder.defineMacro("__ARM_NEON_FP",
756                         "0x" + Twine::utohexstr(HW_FP & ~HW_FP_DP));
757   }
758 
759   if (hasMVE()) {
760     Builder.defineMacro("__ARM_FEATURE_MVE", hasMVEFloat() ? "3" : "1");
761   }
762 
763   Builder.defineMacro("__ARM_SIZEOF_WCHAR_T",
764                       Twine(Opts.WCharSize ? Opts.WCharSize : 4));
765 
766   Builder.defineMacro("__ARM_SIZEOF_MINIMAL_ENUM", Opts.ShortEnums ? "1" : "4");
767 
768   // CMSE
769   if (ArchVersion == 8 && ArchProfile == llvm::ARM::ProfileKind::M)
770     Builder.defineMacro("__ARM_FEATURE_CMSE", Opts.Cmse ? "3" : "1");
771 
772   if (ArchVersion >= 6 && CPUAttr != "6M" && CPUAttr != "8M_BASE") {
773     Builder.defineMacro("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_1");
774     Builder.defineMacro("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_2");
775     Builder.defineMacro("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4");
776     Builder.defineMacro("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_8");
777   }
778 
779   // ACLE 6.4.7 DSP instructions
780   if (DSP) {
781     Builder.defineMacro("__ARM_FEATURE_DSP", "1");
782   }
783 
784   // ACLE 6.4.8 Saturation instructions
785   bool SAT = false;
786   if ((ArchVersion == 6 && CPUProfile != "M") || ArchVersion > 6) {
787     Builder.defineMacro("__ARM_FEATURE_SAT", "1");
788     SAT = true;
789   }
790 
791   // ACLE 6.4.6 Q (saturation) flag
792   if (DSP || SAT)
793     Builder.defineMacro("__ARM_FEATURE_QBIT", "1");
794 
795   if (Opts.UnsafeFPMath)
796     Builder.defineMacro("__ARM_FP_FAST", "1");
797 
798   // Armv8.2-A FP16 vector intrinsic
799   if ((FPU & NeonFPU) && HasLegalHalfType)
800     Builder.defineMacro("__ARM_FEATURE_FP16_VECTOR_ARITHMETIC", "1");
801 
802   // Armv8.2-A FP16 scalar intrinsics
803   if (HasLegalHalfType)
804     Builder.defineMacro("__ARM_FEATURE_FP16_SCALAR_ARITHMETIC", "1");
805 
806   // Armv8.2-A dot product intrinsics
807   if (DotProd)
808     Builder.defineMacro("__ARM_FEATURE_DOTPROD", "1");
809 
810   switch (ArchKind) {
811   default:
812     break;
813   case llvm::ARM::ArchKind::ARMV8_1A:
814     getTargetDefinesARMV81A(Opts, Builder);
815     break;
816   case llvm::ARM::ArchKind::ARMV8_2A:
817     getTargetDefinesARMV82A(Opts, Builder);
818     break;
819   case llvm::ARM::ArchKind::ARMV8_3A:
820   case llvm::ARM::ArchKind::ARMV8_4A:
821   case llvm::ARM::ArchKind::ARMV8_5A:
822     getTargetDefinesARMV83A(Opts, Builder);
823     break;
824   }
825 }
826 
827 const Builtin::Info ARMTargetInfo::BuiltinInfo[] = {
828 #define BUILTIN(ID, TYPE, ATTRS)                                               \
829   {#ID, TYPE, ATTRS, nullptr, ALL_LANGUAGES, nullptr},
830 #define LIBBUILTIN(ID, TYPE, ATTRS, HEADER)                                    \
831   {#ID, TYPE, ATTRS, HEADER, ALL_LANGUAGES, nullptr},
832 #include "clang/Basic/BuiltinsNEON.def"
833 
834 #define BUILTIN(ID, TYPE, ATTRS)                                               \
835   {#ID, TYPE, ATTRS, nullptr, ALL_LANGUAGES, nullptr},
836 #define LANGBUILTIN(ID, TYPE, ATTRS, LANG)                                     \
837   {#ID, TYPE, ATTRS, nullptr, LANG, nullptr},
838 #define LIBBUILTIN(ID, TYPE, ATTRS, HEADER)                                    \
839   {#ID, TYPE, ATTRS, HEADER, ALL_LANGUAGES, nullptr},
840 #define TARGET_HEADER_BUILTIN(ID, TYPE, ATTRS, HEADER, LANGS, FEATURE)         \
841   {#ID, TYPE, ATTRS, HEADER, LANGS, FEATURE},
842 #include "clang/Basic/BuiltinsARM.def"
843 };
844 
845 ArrayRef<Builtin::Info> ARMTargetInfo::getTargetBuiltins() const {
846   return llvm::makeArrayRef(BuiltinInfo, clang::ARM::LastTSBuiltin -
847                                              Builtin::FirstTSBuiltin);
848 }
849 
850 bool ARMTargetInfo::isCLZForZeroUndef() const { return false; }
851 TargetInfo::BuiltinVaListKind ARMTargetInfo::getBuiltinVaListKind() const {
852   return IsAAPCS
853              ? AAPCSABIBuiltinVaList
854              : (getTriple().isWatchABI() ? TargetInfo::CharPtrBuiltinVaList
855                                          : TargetInfo::VoidPtrBuiltinVaList);
856 }
857 
858 const char *const ARMTargetInfo::GCCRegNames[] = {
859     // Integer registers
860     "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11",
861     "r12", "sp", "lr", "pc",
862 
863     // Float registers
864     "s0", "s1", "s2", "s3", "s4", "s5", "s6", "s7", "s8", "s9", "s10", "s11",
865     "s12", "s13", "s14", "s15", "s16", "s17", "s18", "s19", "s20", "s21", "s22",
866     "s23", "s24", "s25", "s26", "s27", "s28", "s29", "s30", "s31",
867 
868     // Double registers
869     "d0", "d1", "d2", "d3", "d4", "d5", "d6", "d7", "d8", "d9", "d10", "d11",
870     "d12", "d13", "d14", "d15", "d16", "d17", "d18", "d19", "d20", "d21", "d22",
871     "d23", "d24", "d25", "d26", "d27", "d28", "d29", "d30", "d31",
872 
873     // Quad registers
874     "q0", "q1", "q2", "q3", "q4", "q5", "q6", "q7", "q8", "q9", "q10", "q11",
875     "q12", "q13", "q14", "q15"};
876 
877 ArrayRef<const char *> ARMTargetInfo::getGCCRegNames() const {
878   return llvm::makeArrayRef(GCCRegNames);
879 }
880 
881 const TargetInfo::GCCRegAlias ARMTargetInfo::GCCRegAliases[] = {
882     {{"a1"}, "r0"},  {{"a2"}, "r1"},        {{"a3"}, "r2"},  {{"a4"}, "r3"},
883     {{"v1"}, "r4"},  {{"v2"}, "r5"},        {{"v3"}, "r6"},  {{"v4"}, "r7"},
884     {{"v5"}, "r8"},  {{"v6", "rfp"}, "r9"}, {{"sl"}, "r10"}, {{"fp"}, "r11"},
885     {{"ip"}, "r12"}, {{"r13"}, "sp"},       {{"r14"}, "lr"}, {{"r15"}, "pc"},
886     // The S, D and Q registers overlap, but aren't really aliases; we
887     // don't want to substitute one of these for a different-sized one.
888 };
889 
890 ArrayRef<TargetInfo::GCCRegAlias> ARMTargetInfo::getGCCRegAliases() const {
891   return llvm::makeArrayRef(GCCRegAliases);
892 }
893 
894 bool ARMTargetInfo::validateAsmConstraint(
895     const char *&Name, TargetInfo::ConstraintInfo &Info) const {
896   switch (*Name) {
897   default:
898     break;
899   case 'l': // r0-r7 if thumb, r0-r15 if ARM
900     Info.setAllowsRegister();
901     return true;
902   case 'h': // r8-r15, thumb only
903     if (isThumb()) {
904       Info.setAllowsRegister();
905       return true;
906     }
907     break;
908   case 's': // An integer constant, but allowing only relocatable values.
909     return true;
910   case 't': // s0-s31, d0-d31, or q0-q15
911   case 'w': // s0-s15, d0-d7, or q0-q3
912   case 'x': // s0-s31, d0-d15, or q0-q7
913     Info.setAllowsRegister();
914     return true;
915   case 'j': // An immediate integer between 0 and 65535 (valid for MOVW)
916     // only available in ARMv6T2 and above
917     if (CPUAttr.equals("6T2") || ArchVersion >= 7) {
918       Info.setRequiresImmediate(0, 65535);
919       return true;
920     }
921     break;
922   case 'I':
923     if (isThumb()) {
924       if (!supportsThumb2())
925         Info.setRequiresImmediate(0, 255);
926       else
927         // FIXME: should check if immediate value would be valid for a Thumb2
928         // data-processing instruction
929         Info.setRequiresImmediate();
930     } else
931       // FIXME: should check if immediate value would be valid for an ARM
932       // data-processing instruction
933       Info.setRequiresImmediate();
934     return true;
935   case 'J':
936     if (isThumb() && !supportsThumb2())
937       Info.setRequiresImmediate(-255, -1);
938     else
939       Info.setRequiresImmediate(-4095, 4095);
940     return true;
941   case 'K':
942     if (isThumb()) {
943       if (!supportsThumb2())
944         // FIXME: should check if immediate value can be obtained from shifting
945         // a value between 0 and 255 left by any amount
946         Info.setRequiresImmediate();
947       else
948         // FIXME: should check if immediate value would be valid for a Thumb2
949         // data-processing instruction when inverted
950         Info.setRequiresImmediate();
951     } else
952       // FIXME: should check if immediate value would be valid for an ARM
953       // data-processing instruction when inverted
954       Info.setRequiresImmediate();
955     return true;
956   case 'L':
957     if (isThumb()) {
958       if (!supportsThumb2())
959         Info.setRequiresImmediate(-7, 7);
960       else
961         // FIXME: should check if immediate value would be valid for a Thumb2
962         // data-processing instruction when negated
963         Info.setRequiresImmediate();
964     } else
965       // FIXME: should check if immediate value  would be valid for an ARM
966       // data-processing instruction when negated
967       Info.setRequiresImmediate();
968     return true;
969   case 'M':
970     if (isThumb() && !supportsThumb2())
971       // FIXME: should check if immediate value is a multiple of 4 between 0 and
972       // 1020
973       Info.setRequiresImmediate();
974     else
975       // FIXME: should check if immediate value is a power of two or a integer
976       // between 0 and 32
977       Info.setRequiresImmediate();
978     return true;
979   case 'N':
980     // Thumb1 only
981     if (isThumb() && !supportsThumb2()) {
982       Info.setRequiresImmediate(0, 31);
983       return true;
984     }
985     break;
986   case 'O':
987     // Thumb1 only
988     if (isThumb() && !supportsThumb2()) {
989       // FIXME: should check if immediate value is a multiple of 4 between -508
990       // and 508
991       Info.setRequiresImmediate();
992       return true;
993     }
994     break;
995   case 'Q': // A memory address that is a single base register.
996     Info.setAllowsMemory();
997     return true;
998   case 'T':
999     switch (Name[1]) {
1000     default:
1001       break;
1002     case 'e': // Even general-purpose register
1003     case 'o': // Odd general-purpose register
1004       Info.setAllowsRegister();
1005       Name++;
1006       return true;
1007     }
1008     break;
1009   case 'U': // a memory reference...
1010     switch (Name[1]) {
1011     case 'q': // ...ARMV4 ldrsb
1012     case 'v': // ...VFP load/store (reg+constant offset)
1013     case 'y': // ...iWMMXt load/store
1014     case 't': // address valid for load/store opaque types wider
1015               // than 128-bits
1016     case 'n': // valid address for Neon doubleword vector load/store
1017     case 'm': // valid address for Neon element and structure load/store
1018     case 's': // valid address for non-offset loads/stores of quad-word
1019               // values in four ARM registers
1020       Info.setAllowsMemory();
1021       Name++;
1022       return true;
1023     }
1024     break;
1025   }
1026   return false;
1027 }
1028 
1029 std::string ARMTargetInfo::convertConstraint(const char *&Constraint) const {
1030   std::string R;
1031   switch (*Constraint) {
1032   case 'U': // Two-character constraint; add "^" hint for later parsing.
1033   case 'T':
1034     R = std::string("^") + std::string(Constraint, 2);
1035     Constraint++;
1036     break;
1037   case 'p': // 'p' should be translated to 'r' by default.
1038     R = std::string("r");
1039     break;
1040   default:
1041     return std::string(1, *Constraint);
1042   }
1043   return R;
1044 }
1045 
1046 bool ARMTargetInfo::validateConstraintModifier(
1047     StringRef Constraint, char Modifier, unsigned Size,
1048     std::string &SuggestedModifier) const {
1049   bool isOutput = (Constraint[0] == '=');
1050   bool isInOut = (Constraint[0] == '+');
1051 
1052   // Strip off constraint modifiers.
1053   while (Constraint[0] == '=' || Constraint[0] == '+' || Constraint[0] == '&')
1054     Constraint = Constraint.substr(1);
1055 
1056   switch (Constraint[0]) {
1057   default:
1058     break;
1059   case 'r': {
1060     switch (Modifier) {
1061     default:
1062       return (isInOut || isOutput || Size <= 64);
1063     case 'q':
1064       // A register of size 32 cannot fit a vector type.
1065       return false;
1066     }
1067   }
1068   }
1069 
1070   return true;
1071 }
1072 const char *ARMTargetInfo::getClobbers() const {
1073   // FIXME: Is this really right?
1074   return "";
1075 }
1076 
1077 TargetInfo::CallingConvCheckResult
1078 ARMTargetInfo::checkCallingConvention(CallingConv CC) const {
1079   switch (CC) {
1080   case CC_AAPCS:
1081   case CC_AAPCS_VFP:
1082   case CC_Swift:
1083   case CC_OpenCLKernel:
1084     return CCCR_OK;
1085   default:
1086     return CCCR_Warning;
1087   }
1088 }
1089 
1090 int ARMTargetInfo::getEHDataRegisterNumber(unsigned RegNo) const {
1091   if (RegNo == 0)
1092     return 0;
1093   if (RegNo == 1)
1094     return 1;
1095   return -1;
1096 }
1097 
1098 bool ARMTargetInfo::hasSjLjLowering() const { return true; }
1099 
1100 ARMleTargetInfo::ARMleTargetInfo(const llvm::Triple &Triple,
1101                                  const TargetOptions &Opts)
1102     : ARMTargetInfo(Triple, Opts) {}
1103 
1104 void ARMleTargetInfo::getTargetDefines(const LangOptions &Opts,
1105                                        MacroBuilder &Builder) const {
1106   Builder.defineMacro("__ARMEL__");
1107   ARMTargetInfo::getTargetDefines(Opts, Builder);
1108 }
1109 
1110 ARMbeTargetInfo::ARMbeTargetInfo(const llvm::Triple &Triple,
1111                                  const TargetOptions &Opts)
1112     : ARMTargetInfo(Triple, Opts) {}
1113 
1114 void ARMbeTargetInfo::getTargetDefines(const LangOptions &Opts,
1115                                        MacroBuilder &Builder) const {
1116   Builder.defineMacro("__ARMEB__");
1117   Builder.defineMacro("__ARM_BIG_ENDIAN");
1118   ARMTargetInfo::getTargetDefines(Opts, Builder);
1119 }
1120 
1121 WindowsARMTargetInfo::WindowsARMTargetInfo(const llvm::Triple &Triple,
1122                                            const TargetOptions &Opts)
1123     : WindowsTargetInfo<ARMleTargetInfo>(Triple, Opts), Triple(Triple) {
1124 }
1125 
1126 void WindowsARMTargetInfo::getVisualStudioDefines(const LangOptions &Opts,
1127                                                   MacroBuilder &Builder) const {
1128   // FIXME: this is invalid for WindowsCE
1129   Builder.defineMacro("_M_ARM_NT", "1");
1130   Builder.defineMacro("_M_ARMT", "_M_ARM");
1131   Builder.defineMacro("_M_THUMB", "_M_ARM");
1132 
1133   assert((Triple.getArch() == llvm::Triple::arm ||
1134           Triple.getArch() == llvm::Triple::thumb) &&
1135          "invalid architecture for Windows ARM target info");
1136   unsigned Offset = Triple.getArch() == llvm::Triple::arm ? 4 : 6;
1137   Builder.defineMacro("_M_ARM", Triple.getArchName().substr(Offset));
1138 
1139   // TODO map the complete set of values
1140   // 31: VFPv3 40: VFPv4
1141   Builder.defineMacro("_M_ARM_FP", "31");
1142 }
1143 
1144 TargetInfo::BuiltinVaListKind
1145 WindowsARMTargetInfo::getBuiltinVaListKind() const {
1146   return TargetInfo::CharPtrBuiltinVaList;
1147 }
1148 
1149 TargetInfo::CallingConvCheckResult
1150 WindowsARMTargetInfo::checkCallingConvention(CallingConv CC) const {
1151   switch (CC) {
1152   case CC_X86StdCall:
1153   case CC_X86ThisCall:
1154   case CC_X86FastCall:
1155   case CC_X86VectorCall:
1156     return CCCR_Ignore;
1157   case CC_C:
1158   case CC_OpenCLKernel:
1159   case CC_PreserveMost:
1160   case CC_PreserveAll:
1161   case CC_Swift:
1162     return CCCR_OK;
1163   default:
1164     return CCCR_Warning;
1165   }
1166 }
1167 
1168 // Windows ARM + Itanium C++ ABI Target
1169 ItaniumWindowsARMleTargetInfo::ItaniumWindowsARMleTargetInfo(
1170     const llvm::Triple &Triple, const TargetOptions &Opts)
1171     : WindowsARMTargetInfo(Triple, Opts) {
1172   TheCXXABI.set(TargetCXXABI::GenericARM);
1173 }
1174 
1175 void ItaniumWindowsARMleTargetInfo::getTargetDefines(
1176     const LangOptions &Opts, MacroBuilder &Builder) const {
1177   WindowsARMTargetInfo::getTargetDefines(Opts, Builder);
1178 
1179   if (Opts.MSVCCompat)
1180     WindowsARMTargetInfo::getVisualStudioDefines(Opts, Builder);
1181 }
1182 
1183 // Windows ARM, MS (C++) ABI
1184 MicrosoftARMleTargetInfo::MicrosoftARMleTargetInfo(const llvm::Triple &Triple,
1185                                                    const TargetOptions &Opts)
1186     : WindowsARMTargetInfo(Triple, Opts) {
1187   TheCXXABI.set(TargetCXXABI::Microsoft);
1188 }
1189 
1190 void MicrosoftARMleTargetInfo::getTargetDefines(const LangOptions &Opts,
1191                                                 MacroBuilder &Builder) const {
1192   WindowsARMTargetInfo::getTargetDefines(Opts, Builder);
1193   WindowsARMTargetInfo::getVisualStudioDefines(Opts, Builder);
1194 }
1195 
1196 MinGWARMTargetInfo::MinGWARMTargetInfo(const llvm::Triple &Triple,
1197                                        const TargetOptions &Opts)
1198     : WindowsARMTargetInfo(Triple, Opts) {
1199   TheCXXABI.set(TargetCXXABI::GenericARM);
1200 }
1201 
1202 void MinGWARMTargetInfo::getTargetDefines(const LangOptions &Opts,
1203                                           MacroBuilder &Builder) const {
1204   WindowsARMTargetInfo::getTargetDefines(Opts, Builder);
1205   Builder.defineMacro("_ARM_");
1206 }
1207 
1208 CygwinARMTargetInfo::CygwinARMTargetInfo(const llvm::Triple &Triple,
1209                                          const TargetOptions &Opts)
1210     : ARMleTargetInfo(Triple, Opts) {
1211   this->WCharType = TargetInfo::UnsignedShort;
1212   TLSSupported = false;
1213   DoubleAlign = LongLongAlign = 64;
1214   resetDataLayout("e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64");
1215 }
1216 
1217 void CygwinARMTargetInfo::getTargetDefines(const LangOptions &Opts,
1218                                            MacroBuilder &Builder) const {
1219   ARMleTargetInfo::getTargetDefines(Opts, Builder);
1220   Builder.defineMacro("_ARM_");
1221   Builder.defineMacro("__CYGWIN__");
1222   Builder.defineMacro("__CYGWIN32__");
1223   DefineStd(Builder, "unix", Opts);
1224   if (Opts.CPlusPlus)
1225     Builder.defineMacro("_GNU_SOURCE");
1226 }
1227 
1228 DarwinARMTargetInfo::DarwinARMTargetInfo(const llvm::Triple &Triple,
1229                                          const TargetOptions &Opts)
1230     : DarwinTargetInfo<ARMleTargetInfo>(Triple, Opts) {
1231   HasAlignMac68kSupport = true;
1232   // iOS always has 64-bit atomic instructions.
1233   // FIXME: This should be based off of the target features in
1234   // ARMleTargetInfo.
1235   MaxAtomicInlineWidth = 64;
1236 
1237   if (Triple.isWatchABI()) {
1238     // Darwin on iOS uses a variant of the ARM C++ ABI.
1239     TheCXXABI.set(TargetCXXABI::WatchOS);
1240 
1241     // BOOL should be a real boolean on the new ABI
1242     UseSignedCharForObjCBool = false;
1243   } else
1244     TheCXXABI.set(TargetCXXABI::iOS);
1245 }
1246 
1247 void DarwinARMTargetInfo::getOSDefines(const LangOptions &Opts,
1248                                        const llvm::Triple &Triple,
1249                                        MacroBuilder &Builder) const {
1250   getDarwinDefines(Builder, Opts, Triple, PlatformName, PlatformMinVersion);
1251 }
1252 
1253 RenderScript32TargetInfo::RenderScript32TargetInfo(const llvm::Triple &Triple,
1254                                                    const TargetOptions &Opts)
1255     : ARMleTargetInfo(llvm::Triple("armv7", Triple.getVendorName(),
1256                                    Triple.getOSName(),
1257                                    Triple.getEnvironmentName()),
1258                       Opts) {
1259   IsRenderScriptTarget = true;
1260   LongWidth = LongAlign = 64;
1261 }
1262 
1263 void RenderScript32TargetInfo::getTargetDefines(const LangOptions &Opts,
1264                                                 MacroBuilder &Builder) const {
1265   Builder.defineMacro("__RENDERSCRIPT__");
1266   ARMleTargetInfo::getTargetDefines(Opts, Builder);
1267 }
1268