12#include "llvm/ADT/StringSwitch.h"
13#include "llvm/Option/ArgList.h"
14#include "llvm/TargetParser/ARMTargetParser.h"
15#include "llvm/TargetParser/Host.h"
24 llvm::StringRef
Arch = Triple.getArchName();
25 return llvm::ARM::parseArchVersion(
Arch);
30 llvm::StringRef
Arch = Triple.getArchName();
31 return llvm::ARM::parseArchProfile(
Arch) == llvm::ARM::ProfileKind::M;
39 if (Arg *A = Args.getLastArg(options::OPT_mlittle_endian,
40 options::OPT_mbig_endian)) {
41 return !A->getOption().matches(options::OPT_mlittle_endian);
44 return Triple.getArch() == llvm::Triple::armeb ||
45 Triple.getArch() == llvm::Triple::thumbeb;
50 llvm::StringRef
Arch = Triple.getArchName();
51 return llvm::ARM::parseArchProfile(
Arch) == llvm::ARM::ProfileKind::A;
56 auto arch = Triple.getArch();
57 if (arch != llvm::Triple::arm && arch != llvm::Triple::thumb &&
58 arch != llvm::Triple::armeb && arch != llvm::Triple::thumbeb)
61 if (Triple.getVendor() != llvm::Triple::UnknownVendor)
64 if (Triple.getOS() != llvm::Triple::UnknownOS)
67 if (Triple.getEnvironment() != llvm::Triple::EABI &&
68 Triple.getEnvironment() != llvm::Triple::EABIHF)
76 llvm::StringRef &CPU,
bool FromAs) {
77 if (
const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ))
79 if (
const Arg *A = Args.getLastArg(options::OPT_march_EQ))
85 Args.filtered(options::OPT_Wa_COMMA, options::OPT_Xassembler)) {
88 for (StringRef
Value : A->getValues()) {
89 if (
Value.starts_with(
"-mcpu="))
90 CPU =
Value.substr(6);
91 if (
Value.starts_with(
"-march="))
100 const ArgList &Args, StringRef HWDiv,
101 std::vector<StringRef> &Features) {
102 uint64_t HWDivID = llvm::ARM::parseHWDiv(HWDiv);
103 if (!llvm::ARM::getHWDivFeatures(HWDivID, Features))
104 D.
Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args);
109 const ArgList &Args, StringRef FPU,
110 std::vector<StringRef> &Features) {
111 llvm::ARM::FPUKind FPUKind = llvm::ARM::parseFPU(FPU);
112 if (!llvm::ARM::getFPUFeatures(FPUKind, Features))
113 D.
Diag(clang::diag::err_drv_clang_unsupported) << A->getAsString(Args);
119 llvm::ARM::ArchKind ArchKind,
120 std::vector<StringRef> &Features,
121 llvm::ARM::FPUKind &ArgFPUKind) {
123 text.split(
Split, StringRef(
"+"), -1,
false);
126 if (!appendArchExtFeatures(CPU, ArchKind,
Feature, Features, ArgFPUKind))
133 std::vector<StringRef> &Features) {
134 CPU = CPU.split(
"+").first;
135 if (CPU !=
"generic") {
136 llvm::ARM::ArchKind ArchKind = llvm::ARM::parseCPUArch(CPU);
137 uint64_t Extension = llvm::ARM::getDefaultExtensions(CPU, ArchKind);
138 llvm::ARM::getExtensionFeatures(Extension, Features);
146 llvm::StringRef ArchName, llvm::StringRef CPUName,
147 std::vector<StringRef> &Features,
148 const llvm::Triple &Triple,
149 llvm::ARM::FPUKind &ArgFPUKind) {
150 std::pair<StringRef, StringRef>
Split = ArchName.split(
"+");
153 llvm::ARM::ArchKind ArchKind = llvm::ARM::parseArch(MArch);
154 if (ArchKind == llvm::ARM::ArchKind::INVALID ||
155 (
Split.second.size() &&
158 D.
Diag(clang::diag::err_drv_unsupported_option_argument)
159 << A->getSpelling() << A->getValue();
164 llvm::StringRef CPUName, llvm::StringRef ArchName,
165 std::vector<StringRef> &Features,
166 const llvm::Triple &Triple,
167 llvm::ARM::FPUKind &ArgFPUKind) {
168 std::pair<StringRef, StringRef>
Split = CPUName.split(
"+");
171 llvm::ARM::ArchKind ArchKind =
173 if (ArchKind == llvm::ARM::ArchKind::INVALID ||
175 Features, ArgFPUKind)))
176 D.
Diag(clang::diag::err_drv_unsupported_option_argument)
177 << A->getSpelling() << A->getValue();
187 Args.getLastArg(options::OPT_msoft_float, options::OPT_mhard_float,
188 options::OPT_mfloat_abi_EQ);
189 if (A && (A->getOption().matches(options::OPT_mhard_float) ||
190 (A->getOption().matches(options::OPT_mfloat_abi_EQ) &&
191 A->getValue() == StringRef(
"hard"))))
192 D.
Diag(clang::diag::warn_drv_no_floating_point_registers)
193 << A->getAsString(Args);
199 return T.getEnvironment() == llvm::Triple::EABI ||
200 T.getEnvironment() == llvm::Triple::EABIHF ||
212 llvm::ARM::ArchKind AK = llvm::ARM::parseArch(Triple.getArchName());
213 return AK == llvm::ARM::ArchKind::ARMV6K ||
214 AK == llvm::ARM::ArchKind::ARMV6KZ ||
221 llvm::ARM::ArchKind AK = llvm::ARM::parseArch(Triple.getArchName());
222 return AK == llvm::ARM::ArchKind::ARMV6T2 ||
223 (Ver >= 7 && AK != llvm::ARM::ArchKind::ARMV8MBaseline);
228 const llvm::Triple &Triple,
bool ForAS) {
229 Arg *A = Args.getLastArg(options::OPT_mtp_mode_EQ);
230 if (A && A->getValue() != StringRef(
"auto")) {
232 llvm::StringSwitch<arm::ReadTPMode>(A->getValue())
233 .Case(
"cp15", ReadTPMode::TPIDRURO)
234 .Case(
"tpidrurw", ReadTPMode::TPIDRURW)
235 .Case(
"tpidruro", ReadTPMode::TPIDRURO)
236 .Case(
"tpidrprw", ReadTPMode::TPIDRPRW)
237 .Case(
"soft", ReadTPMode::Soft)
238 .Default(ReadTPMode::Invalid);
239 if ((ThreadPointer == ReadTPMode::TPIDRURW ||
240 ThreadPointer == ReadTPMode::TPIDRURO ||
241 ThreadPointer == ReadTPMode::TPIDRPRW) &&
243 D.
Diag(diag::err_target_unsupported_tp_hard) << Triple.getArchName();
244 return ReadTPMode::Invalid;
246 if (ThreadPointer != ReadTPMode::Invalid)
247 return ThreadPointer;
248 if (StringRef(A->getValue()).empty())
249 D.
Diag(diag::err_drv_missing_arg_mtp) << A->getAsString(Args);
251 D.
Diag(diag::err_drv_invalid_mtp) << A->getAsString(Args);
252 return ReadTPMode::Invalid;
258 bool autoUseHWTPMode =
260 return autoUseHWTPMode ? ReadTPMode::TPIDRURO : ReadTPMode::Soft;
264 types::ID InputType, llvm::Triple &Triple) {
265 StringRef MCPU, MArch;
266 if (
const Arg *A = Args.getLastArg(options::OPT_mcpu_EQ))
267 MCPU = A->getValue();
268 if (
const Arg *A = Args.getLastArg(options::OPT_march_EQ))
269 MArch = A->getValue();
271 std::string
CPU = Triple.isOSBinFormatMachO()
276 bool IsBigEndian = Triple.getArch() == llvm::Triple::armeb ||
277 Triple.getArch() == llvm::Triple::thumbeb;
280 if (Arg *A = Args.getLastArg(options::OPT_mlittle_endian,
281 options::OPT_mbig_endian)) {
282 IsBigEndian = !A->getOption().matches(options::OPT_mlittle_endian);
284 std::string ArchName = IsBigEndian ?
"armeb" :
"arm";
288 llvm::ARM::parseArchProfile(Suffix) == llvm::ARM::ProfileKind::M;
289 bool ThumbDefault = IsMProfile ||
291 (llvm::ARM::parseArchVersion(Suffix) == 7 &&
292 Triple.isOSBinFormatMachO()) ||
294 Triple.isOSFuchsia() ||
296 Triple.isOSWindows();
300 bool ARMModeRequested =
301 !Args.hasFlag(options::OPT_mthumb, options::OPT_mno_thumb, ThumbDefault);
302 if (IsMProfile && ARMModeRequested) {
304 D.
Diag(diag::err_cpu_unsupported_isa) <<
CPU <<
"ARM";
306 D.
Diag(diag::err_arch_unsupported_isa)
313 bool IsThumb =
false;
314 if (InputType != types::TY_PP_Asm)
316 Args.hasFlag(options::OPT_mthumb, options::OPT_mno_thumb, ThumbDefault);
321 llvm::StringRef WaMArch, WaMCPU;
323 Args.filtered(options::OPT_Wa_COMMA, options::OPT_Xassembler)) {
324 for (StringRef
Value : A->getValues()) {
326 if (
Value ==
"-mthumb")
328 else if (
Value.starts_with(
"-march="))
329 WaMArch =
Value.substr(7);
330 else if (
Value.starts_with(
"-mcpu="))
331 WaMCPU =
Value.substr(6);
335 if (WaMCPU.size() || WaMArch.size()) {
345 if (IsThumb || IsMProfile || Triple.isOSWindows()) {
347 ArchName =
"thumbeb";
351 Triple.setArchName(ArchName + Suffix.str());
355 llvm::Triple &Triple) {
356 if (Triple.isOSLiteOS()) {
357 Triple.setEnvironment(llvm::Triple::OpenHOS);
364 switch (Triple.getEnvironment()) {
365 case llvm::Triple::GNUEABI:
366 case llvm::Triple::GNUEABIHF:
367 Triple.setEnvironment(isHardFloat ? llvm::Triple::GNUEABIHF
368 : llvm::Triple::GNUEABI);
370 case llvm::Triple::GNUEABIT64:
371 case llvm::Triple::GNUEABIHFT64:
372 Triple.setEnvironment(isHardFloat ? llvm::Triple::GNUEABIHFT64
373 : llvm::Triple::GNUEABIT64);
375 case llvm::Triple::EABI:
376 case llvm::Triple::EABIHF:
377 Triple.setEnvironment(isHardFloat ? llvm::Triple::EABIHF
378 : llvm::Triple::EABI);
380 case llvm::Triple::MuslEABI:
381 case llvm::Triple::MuslEABIHF:
382 Triple.setEnvironment(isHardFloat ? llvm::Triple::MuslEABIHF
383 : llvm::Triple::MuslEABI);
385 case llvm::Triple::OpenHOS:
392 Args.getLastArg(options::OPT_msoft_float, options::OPT_mhard_float,
393 options::OPT_mfloat_abi_EQ);
394 assert(ABIArg &&
"Non-default float abi expected to be from arg");
395 D.
Diag(diag::err_drv_unsupported_opt_for_target)
396 << ABIArg->getAsString(Args) << Triple.getTriple();
409 switch (Triple.getOS()) {
410 case llvm::Triple::Darwin:
411 case llvm::Triple::MacOSX:
412 case llvm::Triple::IOS:
413 case llvm::Triple::TvOS:
414 case llvm::Triple::DriverKit:
415 case llvm::Triple::XROS:
417 if (Triple.isWatchABI())
422 case llvm::Triple::WatchOS:
426 case llvm::Triple::Win32:
433 case llvm::Triple::NetBSD:
434 switch (Triple.getEnvironment()) {
435 case llvm::Triple::EABIHF:
436 case llvm::Triple::GNUEABIHF:
443 case llvm::Triple::FreeBSD:
444 switch (Triple.getEnvironment()) {
445 case llvm::Triple::GNUEABIHF:
453 case llvm::Triple::Haiku:
454 case llvm::Triple::OpenBSD:
457 case llvm::Triple::Fuchsia:
461 if (Triple.isOHOSFamily())
463 switch (Triple.getEnvironment()) {
464 case llvm::Triple::GNUEABIHF:
465 case llvm::Triple::GNUEABIHFT64:
466 case llvm::Triple::MuslEABIHF:
467 case llvm::Triple::EABIHF:
469 case llvm::Triple::Android:
470 case llvm::Triple::GNUEABI:
471 case llvm::Triple::GNUEABIT64:
472 case llvm::Triple::MuslEABI:
473 case llvm::Triple::EABI:
486 const ArgList &Args) {
489 Args.getLastArg(options::OPT_msoft_float, options::OPT_mhard_float,
490 options::OPT_mfloat_abi_EQ)) {
491 if (A->getOption().matches(options::OPT_msoft_float)) {
492 ABI = FloatABI::Soft;
493 }
else if (A->getOption().matches(options::OPT_mhard_float)) {
494 ABI = FloatABI::Hard;
496 ABI = llvm::StringSwitch<arm::FloatABI>(A->getValue())
497 .Case(
"soft", FloatABI::Soft)
498 .Case(
"softfp", FloatABI::SoftFP)
499 .Case(
"hard", FloatABI::Hard)
500 .Default(FloatABI::Invalid);
501 if (ABI == FloatABI::Invalid && !StringRef(A->getValue()).empty()) {
502 D.
Diag(diag::err_drv_invalid_mfloat_abi) << A->getAsString(Args);
503 ABI = FloatABI::Soft;
509 if (ABI == FloatABI::Invalid)
512 if (ABI == FloatABI::Invalid) {
514 if (Triple.isOSBinFormatMachO() &&
515 Triple.getSubArch() == llvm::Triple::ARMSubArch_v7em)
516 ABI = FloatABI::Hard;
518 ABI = FloatABI::Soft;
520 if (((Triple.getOS() != llvm::Triple::UnknownOS) &&
521 !Triple.isOSFirmware()) ||
522 !Triple.isOSBinFormatMachO())
523 D.
Diag(diag::warn_drv_assuming_mfloat_abi_is) <<
"soft";
526 assert(ABI != FloatABI::Invalid &&
"must select an ABI");
531 auto MVE = llvm::find(llvm::reverse(F),
"+mve");
532 auto NoMVE = llvm::find(llvm::reverse(F),
"-mve");
533 return MVE != F.rend() &&
534 (NoMVE == F.rend() || std::distance(
MVE, NoMVE) > 0);
538 const llvm::Triple &Triple,
540 std::vector<StringRef> &Features,
541 bool ForAS,
bool ForMultilib) {
543 Args.hasArg(options::OPT_mkernel, options::OPT_fapple_kext);
545 std::optional<std::pair<const Arg *, StringRef>> WaCPU, WaFPU, WaHDiv, WaArch;
552 std::vector<StringRef> ExtensionFeatures;
569 Features.push_back(
"+soft-float");
573 Features.push_back(
"+soft-float-abi");
578 Args.filtered(options::OPT_Wa_COMMA, options::OPT_Xassembler)) {
581 for (StringRef
Value : A->getValues()) {
582 if (
Value.starts_with(
"-mfpu=")) {
583 WaFPU = std::make_pair(A,
Value.substr(6));
584 }
else if (
Value.starts_with(
"-mcpu=")) {
585 WaCPU = std::make_pair(A,
Value.substr(6));
586 }
else if (
Value.starts_with(
"-mhwdiv=")) {
587 WaHDiv = std::make_pair(A,
Value.substr(8));
588 }
else if (
Value.starts_with(
"-march=")) {
589 WaArch = std::make_pair(A,
Value.substr(7));
597 if (Arg *A = Args.getLastArgNoClaim(options::OPT_mabi_EQ))
598 A->ignoreTargetSpecific();
603 if (TPMode == ReadTPMode::TPIDRURW)
604 Features.push_back(
"+read-tp-tpidrurw");
605 else if (TPMode == ReadTPMode::TPIDRPRW)
606 Features.push_back(
"+read-tp-tpidrprw");
607 else if (TPMode == ReadTPMode::TPIDRURO)
608 Features.push_back(
"+read-tp-tpidruro");
610 const Arg *ArchArg = Args.getLastArg(options::OPT_march_EQ);
611 const Arg *CPUArg = Args.getLastArg(options::OPT_mcpu_EQ);
614 llvm::ARM::FPUKind ArchArgFPUKind = llvm::ARM::FK_INVALID;
615 llvm::ARM::FPUKind CPUArgFPUKind = llvm::ARM::FK_INVALID;
620 D.
Diag(clang::diag::warn_drv_unused_argument)
621 << CPUArg->getAsString(Args);
622 CPUName = WaCPU->second;
623 CPUArg = WaCPU->first;
625 CPUName = CPUArg->getValue();
630 D.
Diag(clang::diag::warn_drv_unused_argument)
631 << ArchArg->getAsString(Args);
632 ArchName = WaArch->second;
635 ExtensionFeatures, Triple, ArchArgFPUKind);
638 }
else if (ArchArg) {
639 ArchName = ArchArg->getValue();
641 Triple, ArchArgFPUKind);
645 if (CPUName ==
"native") {
646 for (
auto &F : llvm::sys::getHostCPUFeatures())
648 Args.MakeArgString((F.second ?
"+" :
"-") + F.first()));
649 }
else if (!CPUName.empty()) {
659 Triple, CPUArgFPUKind);
663 (void)Args.getLastArg(options::OPT_mtune_EQ);
666 llvm::ARM::FPUKind FPUKind = llvm::ARM::FK_INVALID;
667 const Arg *FPUArg = Args.getLastArg(options::OPT_mfpu_EQ);
670 D.
Diag(clang::diag::warn_drv_unused_argument)
671 << FPUArg->getAsString(Args);
676 const char *AndroidFPU =
"neon";
677 FPUKind = llvm::ARM::parseFPU(AndroidFPU);
678 if (!llvm::ARM::getFPUFeatures(FPUKind, Features))
679 D.
Diag(clang::diag::err_drv_clang_unsupported)
680 << std::string(
"-mfpu=") + AndroidFPU;
681 }
else if (ArchArgFPUKind != llvm::ARM::FK_INVALID ||
682 CPUArgFPUKind != llvm::ARM::FK_INVALID) {
684 CPUArgFPUKind != llvm::ARM::FK_INVALID ? CPUArgFPUKind : ArchArgFPUKind;
685 (void)llvm::ARM::getFPUFeatures(FPUKind, Features);
689 if (
Generic && (Triple.isOSWindows() || Triple.isOSDarwin()) &&
691 FPUKind = llvm::ARM::parseFPU(
"neon");
693 llvm::ARM::ArchKind ArchKind =
695 FPUKind = llvm::ARM::getDefaultFPU(CPU, ArchKind);
697 (void)llvm::ARM::getFPUFeatures(FPUKind, Features);
703 Features.insert(std::end(Features),
704 std::begin(ExtensionFeatures), std::end(ExtensionFeatures));
707 const Arg *HDivArg = Args.getLastArg(options::OPT_mhwdiv_EQ);
710 D.
Diag(clang::diag::warn_drv_unused_argument)
711 << HDivArg->getAsString(Args);
720 const auto ItRNoFullFP16 = std::find(Features.rbegin(), Features.rend(),
"-fullfp16");
721 const auto ItRFP16FML = std::find(Features.rbegin(), Features.rend(),
"+fp16fml");
722 if (Triple.getSubArch() == llvm::Triple::SubArchType::ARMSubArch_v8_4a) {
723 const auto ItRFullFP16 = std::find(Features.rbegin(), Features.rend(),
"+fullfp16");
724 if (ItRFullFP16 < ItRNoFullFP16 && ItRFullFP16 < ItRFP16FML) {
727 if (std::find(Features.rbegin(), ItRFullFP16,
"-fp16fml") == ItRFullFP16)
728 Features.push_back(
"+fp16fml");
731 goto fp16_fml_fallthrough;
737 if (ItRNoFullFP16 < ItRFP16FML)
738 Features.push_back(
"-fp16fml");
739 else if (ItRNoFullFP16 > ItRFP16FML)
740 Features.push_back(
"+fullfp16");
747 bool HasFPRegs =
true;
749 llvm::ARM::getFPUFeatures(llvm::ARM::FK_NONE, Features);
753 Features.insert(Features.end(),
754 {
"-dotprod",
"-fp16fml",
"-bf16",
"-mve",
"-mve.fp"});
756 FPUKind = llvm::ARM::FK_NONE;
757 }
else if (FPUKind == llvm::ARM::FK_NONE ||
758 ArchArgFPUKind == llvm::ARM::FK_NONE ||
759 CPUArgFPUKind == llvm::ARM::FK_NONE) {
764 Features.insert(Features.end(),
765 {
"-dotprod",
"-fp16fml",
"-bf16",
"-mve.fp"});
767 FPUKind = llvm::ARM::FK_NONE;
770 Features.emplace_back(
"-fpregs");
773 if (Arg *A = Args.getLastArg(options::OPT_mcrc, options::OPT_mnocrc)) {
774 if (A->getOption().matches(options::OPT_mcrc))
775 Features.push_back(
"+crc");
777 Features.push_back(
"-crc");
786 if (FPUKind == llvm::ARM::FK_FPV5_D16 || FPUKind == llvm::ARM::FK_FPV5_SP_D16)
787 Features.push_back(
"-mve.fp");
791 bool HasSimd =
false;
793 llvm::find_if(llvm::reverse(Features),
794 [](
const StringRef F) {
return F.contains(
"neon"); });
795 const bool FPUSupportsNeon = (llvm::ARM::FPUNames[FPUKind].NeonSupport ==
796 llvm::ARM::NeonSupportLevel::Neon) ||
797 (llvm::ARM::FPUNames[FPUKind].NeonSupport ==
798 llvm::ARM::NeonSupportLevel::Crypto);
799 if (ItSimd != Features.rend())
800 HasSimd = ItSimd->starts_with(
"+");
801 if (!HasSimd && FPUSupportsNeon)
802 Features.insert(Features.end(),
803 {
"-sha2",
"-aes",
"-crypto",
"-dotprod",
"-bf16",
"-i8mm"});
813 bool HasSHA2 =
false;
815 bool HasBF16 =
false;
816 bool HasDotprod =
false;
817 bool HasI8MM =
false;
818 const auto ItCrypto =
819 llvm::find_if(llvm::reverse(Features), [](
const StringRef F) {
820 return F.contains(
"crypto");
823 llvm::find_if(llvm::reverse(Features), [](
const StringRef F) {
824 return F.contains(
"crypto") || F.contains(
"sha2");
827 llvm::find_if(llvm::reverse(Features), [](
const StringRef F) {
828 return F.contains(
"crypto") || F.contains(
"aes");
831 llvm::find_if(llvm::reverse(Features),
832 [](
const StringRef F) {
return F.contains(
"bf16"); });
833 const auto ItDotprod =
834 llvm::find_if(llvm::reverse(Features),
835 [](
const StringRef F) {
return F.contains(
"dotprod"); });
837 llvm::find_if(llvm::reverse(Features),
838 [](
const StringRef F) {
return F.contains(
"i8mm"); });
839 if (ItSHA2 != Features.rend())
840 HasSHA2 = ItSHA2->starts_with(
"+");
841 if (ItAES != Features.rend())
842 HasAES = ItAES->starts_with(
"+");
843 if (ItBF16 != Features.rend())
844 HasBF16 = ItBF16->starts_with(
"+");
845 if (ItDotprod != Features.rend())
846 HasDotprod = ItDotprod->starts_with(
"+");
847 if (ItI8MM != Features.rend())
848 HasI8MM = ItI8MM->starts_with(
"+");
849 if (ItCrypto != Features.rend()) {
850 if (HasSHA2 && HasAES)
851 Features.push_back(
"+crypto");
853 Features.push_back(
"-crypto");
855 Features.push_back(
"+sha2");
857 Features.push_back(
"-sha2");
859 Features.push_back(
"+aes");
861 Features.push_back(
"-aes");
864 if (HasAES || HasSHA2 || HasBF16 || HasDotprod || HasI8MM)
865 Features.push_back(
"+neon");
867 if (HasSHA2 || HasAES) {
870 llvm::ARM::ProfileKind ArchProfile =
871 llvm::ARM::parseArchProfile(ArchSuffix);
872 if (!((llvm::ARM::parseArchVersion(ArchSuffix) >= 8) &&
873 (ArchProfile == llvm::ARM::ProfileKind::A ||
874 ArchProfile == llvm::ARM::ProfileKind::R))) {
876 D.
Diag(clang::diag::warn_target_unsupported_extension)
878 << llvm::ARM::getArchName(llvm::ARM::parseArch(ArchSuffix));
880 D.
Diag(clang::diag::warn_target_unsupported_extension)
882 << llvm::ARM::getArchName(llvm::ARM::parseArch(ArchSuffix));
888 if (!Args.hasArg(options::OPT_fno_integrated_as)) {
889 Features.push_back(
"-sha2");
890 Features.push_back(
"-aes");
896 if (Arg *A = Args.getLastArg(options::OPT_mframe_chain)) {
897 StringRef FrameChainOption = A->getValue();
898 if (FrameChainOption.starts_with(
"aapcs"))
899 Features.push_back(
"+aapcs-frame-chain");
903 if (Args.getLastArg(options::OPT_mcmse))
904 Features.push_back(
"+8msecext");
906 if (Arg *A = Args.getLastArg(options::OPT_mfix_cmse_cve_2021_35465,
907 options::OPT_mno_fix_cmse_cve_2021_35465)) {
908 if (!Args.getLastArg(options::OPT_mcmse))
909 D.
Diag(diag::err_opt_not_valid_without_opt)
910 << A->getOption().getName() <<
"-mcmse";
912 if (A->getOption().matches(options::OPT_mfix_cmse_cve_2021_35465))
913 Features.push_back(
"+fix-cmse-cve-2021-35465");
915 Features.push_back(
"-fix-cmse-cve-2021-35465");
919 if (Arg *A = Args.getLastArg(options::OPT_mfix_cortex_a57_aes_1742098,
920 options::OPT_mno_fix_cortex_a57_aes_1742098)) {
921 if (A->getOption().matches(options::OPT_mfix_cortex_a57_aes_1742098)) {
922 Features.push_back(
"+fix-cortex-a57-aes-1742098");
924 Features.push_back(
"-fix-cortex-a57-aes-1742098");
931 if (Arg *A = Args.getLastArg(options::OPT_mlong_calls,
932 options::OPT_mno_long_calls)) {
933 if (A->getOption().matches(options::OPT_mlong_calls))
934 Features.push_back(
"+long-calls");
935 }
else if (KernelOrKext && (!Triple.isiOS() || Triple.isOSVersionLT(6)) &&
936 !Triple.isWatchOS() && !Triple.isXROS()) {
937 Features.push_back(
"+long-calls");
944 if (!ForAS && !ForMultilib) {
947 if (Arg *A = Args.getLastArg(options::OPT_mexecute_only, options::OPT_mno_execute_only)) {
948 if (A->getOption().matches(options::OPT_mexecute_only)) {
950 llvm::ARM::parseArch(Triple.getArchName()) != llvm::ARM::ArchKind::ARMV6T2 &&
951 llvm::ARM::parseArch(Triple.getArchName()) != llvm::ARM::ArchKind::ARMV6M)
952 D.
Diag(diag::err_target_unsupported_execute_only) << Triple.getArchName();
953 else if (llvm::ARM::parseArch(Triple.getArchName()) == llvm::ARM::ArchKind::ARMV6M) {
954 if (Arg *PIArg = Args.getLastArg(options::OPT_fropi, options::OPT_frwpi,
955 options::OPT_fpic, options::OPT_fpie,
956 options::OPT_fPIC, options::OPT_fPIE))
957 D.
Diag(diag::err_opt_not_valid_with_opt_on_target)
958 << A->getAsString(Args) << PIArg->getAsString(Args) << Triple.getArchName();
959 }
else if (Arg *B = Args.getLastArg(options::OPT_mno_movt))
960 D.
Diag(diag::err_opt_not_valid_with_opt)
961 << A->getAsString(Args) << B->getAsString(Args);
962 Features.push_back(
"+execute-only");
967 if (Arg *A = Args.getLastArg(options::OPT_mno_unaligned_access,
968 options::OPT_munaligned_access,
969 options::OPT_mstrict_align,
970 options::OPT_mno_strict_align)) {
973 A->getOption().matches(options::OPT_mno_unaligned_access) ||
974 A->getOption().matches(options::OPT_mstrict_align)) {
975 Features.push_back(
"+strict-align");
978 if (Triple.getSubArch() == llvm::Triple::SubArchType::ARMSubArch_v6m)
979 D.
Diag(diag::err_target_unsupported_unaligned) <<
"v6m";
982 else if (Triple.getSubArch() == llvm::Triple::SubArchType::ARMSubArch_v8m_baseline)
983 D.
Diag(diag::err_target_unsupported_unaligned) <<
"v8m.base";
1000 if (Triple.isOSDarwin() || Triple.isOSNetBSD()) {
1001 if (VersionNum < 6 ||
1002 Triple.getSubArch() == llvm::Triple::SubArchType::ARMSubArch_v6m)
1003 Features.push_back(
"+strict-align");
1004 }
else if (Triple.getVendor() == llvm::Triple::Apple &&
1005 Triple.isOSBinFormatMachO()) {
1007 Features.push_back(
"+strict-align");
1008 }
else if (VersionNum < 7 ||
1009 Triple.getSubArch() ==
1010 llvm::Triple::SubArchType::ARMSubArch_v6m ||
1011 Triple.getSubArch() ==
1012 llvm::Triple::SubArchType::ARMSubArch_v8m_baseline) {
1013 Features.push_back(
"+strict-align");
1020 if (Args.hasArg(options::OPT_ffixed_r9))
1021 Features.push_back(
"+reserve-r9");
1024 if (KernelOrKext || Args.hasArg(options::OPT_mno_movt))
1025 Features.push_back(
"+no-movt");
1027 if (Args.hasArg(options::OPT_mno_neg_immediates))
1028 Features.push_back(
"+no-neg-immediates");
1031 if (Arg *A = Args.getLastArg(options::OPT_mharden_sls_EQ)) {
1032 StringRef
Scope = A->getValue();
1033 bool EnableRetBr =
false;
1034 bool EnableBlr =
false;
1035 bool DisableComdat =
false;
1036 if (
Scope !=
"none") {
1038 Scope.split(Opts,
",");
1039 for (
auto Opt : Opts) {
1046 if (Opt ==
"retbr") {
1054 if (Opt ==
"comdat") {
1055 DisableComdat =
false;
1058 if (Opt ==
"nocomdat") {
1059 DisableComdat =
true;
1062 D.
Diag(diag::err_drv_unsupported_option_argument)
1063 << A->getSpelling() <<
Scope;
1068 if (EnableRetBr || EnableBlr)
1070 D.
Diag(diag::err_sls_hardening_arm_not_supported)
1071 <<
Scope << A->getAsString(Args);
1074 Features.push_back(
"+harden-sls-retbr");
1076 Features.push_back(
"+harden-sls-blr");
1077 if (DisableComdat) {
1078 Features.push_back(
"+harden-sls-nocomdat");
1082 if (Args.getLastArg(options::OPT_mno_bti_at_return_twice))
1083 Features.push_back(
"+no-bti-at-return-twice");
1093 MArch = std::string(
Arch);
1095 MArch = std::string(Triple.getArchName());
1096 MArch = StringRef(MArch).split(
"+").first.lower();
1099 if (MArch ==
"native") {
1100 std::string
CPU = std::string(llvm::sys::getHostCPUName());
1101 if (CPU !=
"generic") {
1109 MArch = std::string(
"arm") + Suffix.str();
1126 return llvm::ARM::getARMCPUForArch(Triple, MArch);
1131 const llvm::Triple &Triple) {
1135 std::string MCPU = StringRef(CPU).split(
"+").first.lower();
1137 if (MCPU ==
"native")
1138 return std::string(llvm::sys::getHostCPUName());
1151 const llvm::Triple &Triple) {
1152 llvm::ARM::ArchKind ArchKind;
1153 if (CPU ==
"generic" || CPU.empty()) {
1155 ArchKind = llvm::ARM::parseArch(ARMArch);
1156 if (ArchKind == llvm::ARM::ArchKind::INVALID)
1160 llvm::ARM::parseCPUArch(llvm::ARM::getARMCPUForArch(Triple, ARMArch));
1164 ArchKind = (
Arch ==
"armv7k" ||
Arch ==
"thumbv7k")
1165 ? llvm::ARM::ArchKind::ARMV7K
1166 : llvm::ARM::parseCPUArch(CPU);
1175 const llvm::Triple &Triple) {
1176 llvm::ARM::ArchKind ArchKind = getLLVMArchKindForARM(CPU,
Arch, Triple);
1177 if (ArchKind == llvm::ARM::ArchKind::INVALID)
1179 return llvm::ARM::getSubArch(ArchKind);
1183 const llvm::Triple &Triple) {
1184 if (Args.hasArg(options::OPT_r))
1190 CmdArgs.push_back(
"--be8");
Scope - A scope is a transient data structure that is used while parsing the program.
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
DiagnosticBuilder Diag(unsigned DiagID) const
The JSON file list parser is used to communicate input to InstallAPI.
@ Generic
not a target-specific vector type