//===-- AArch64TargetParser - Parser for AArch64 features -------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // This file implements a target parser to recognise AArch64 hardware features // such as FPU/CPU/ARCH and extension names. // //===----------------------------------------------------------------------===// #include "llvm/Support/AArch64TargetParser.h" #include "llvm/ADT/StringSwitch.h" #include "llvm/ADT/Triple.h" #include "llvm/Support/ARMTargetParserCommon.h" #include using namespace llvm; static unsigned checkArchVersion(llvm::StringRef Arch) { if (Arch.size() >= 2 && Arch[0] == 'v' && std::isdigit(Arch[1])) return (Arch[1] - 48); return 0; } uint64_t AArch64::getDefaultExtensions(StringRef CPU, AArch64::ArchKind AK) { if (CPU == "generic") return AArch64ARCHNames[static_cast(AK)].ArchBaseExtensions; return StringSwitch(CPU) #define AARCH64_CPU_NAME(NAME, ID, DEFAULT_FPU, IS_DEFAULT, DEFAULT_EXT) \ .Case(NAME, AArch64ARCHNames[static_cast(ArchKind::ID)] \ .ArchBaseExtensions | \ DEFAULT_EXT) #include "../../include/llvm/Support/AArch64TargetParser.def" .Default(AArch64::AEK_INVALID); } AArch64::ArchKind AArch64::getCPUArchKind(StringRef CPU) { if (CPU == "generic") return ArchKind::ARMV8A; return StringSwitch(CPU) #define AARCH64_CPU_NAME(NAME, ID, DEFAULT_FPU, IS_DEFAULT, DEFAULT_EXT) \ .Case(NAME, ArchKind::ID) #include "../../include/llvm/Support/AArch64TargetParser.def" .Default(ArchKind::INVALID); } AArch64::ArchKind AArch64::getSubArchArchKind(StringRef SubArch) { for (const auto &A : AArch64ARCHNames) if (A.getSubArch() == SubArch) return A.ID; return ArchKind::INVALID; } bool AArch64::getExtensionFeatures(uint64_t Extensions, std::vector &Features) { if (Extensions == AArch64::AEK_INVALID) return false; #define AARCH64_ARCH_EXT_NAME(NAME, ID, FEATURE, NEGFEATURE) \ if (Extensions & ID) { \ const char *feature = FEATURE; \ /* INVALID and NONE have no feature name. */ \ if (feature) \ Features.push_back(feature); \ } #include "../../include/llvm/Support/AArch64TargetParser.def" return true; } StringRef AArch64::resolveCPUAlias(StringRef CPU) { return StringSwitch(CPU) #define AARCH64_CPU_ALIAS(ALIAS,NAME) \ .Case(ALIAS, NAME) #include "../../include/llvm/Support/AArch64TargetParser.def" .Default(CPU); } bool AArch64::getArchFeatures(AArch64::ArchKind AK, std::vector &Features) { if (AK == ArchKind::INVALID) return false; Features.push_back(AArch64ARCHNames[static_cast(AK)].ArchFeature); return true; } StringRef AArch64::getArchName(AArch64::ArchKind AK) { return AArch64ARCHNames[static_cast(AK)].Name; } StringRef AArch64::getSubArch(AArch64::ArchKind AK) { return AArch64ARCHNames[static_cast(AK)].getSubArch(); } StringRef AArch64::getArchExtFeature(StringRef ArchExt) { if (ArchExt.startswith("no")) { StringRef ArchExtBase(ArchExt.substr(2)); for (const auto &AE : AArch64ARCHExtNames) { if (!AE.NegFeature.empty() && ArchExtBase == AE.Name) return AE.NegFeature; } } for (const auto &AE : AArch64ARCHExtNames) if (!AE.Feature.empty() && ArchExt == AE.Name) return AE.Feature; return StringRef(); } AArch64::ArchKind AArch64::convertV9toV8(AArch64::ArchKind AK) { if (AK == AArch64::ArchKind::INVALID) return AK; if (AK < AArch64::ArchKind::ARMV9A) return AK; if (AK >= AArch64::ArchKind::ARMV8R) return AArch64::ArchKind::INVALID; unsigned AK_v8 = static_cast(AArch64::ArchKind::ARMV8_5A); AK_v8 += static_cast(AK) - static_cast(AArch64::ArchKind::ARMV9A); return static_cast(AK_v8); } void AArch64::fillValidCPUArchList(SmallVectorImpl &Values) { for (const auto &Arch : AArch64CPUNames) { if (Arch.ArchID != ArchKind::INVALID) Values.push_back(Arch.Name); } for (const auto &Alias: AArch64CPUAliases) Values.push_back(Alias.Alias); } bool AArch64::isX18ReservedByDefault(const Triple &TT) { return TT.isAndroid() || TT.isOSDarwin() || TT.isOSFuchsia() || TT.isOSWindows(); } // Allows partial match, ex. "v8a" matches "armv8a". AArch64::ArchKind AArch64::parseArch(StringRef Arch) { Arch = llvm::ARM::getCanonicalArchName(Arch); if (checkArchVersion(Arch) < 8) return ArchKind::INVALID; StringRef Syn = llvm::ARM::getArchSynonym(Arch); for (const auto &A : AArch64ARCHNames) { if (A.Name.endswith(Syn)) return A.ID; } return ArchKind::INVALID; } AArch64::ArchExtKind AArch64::parseArchExt(StringRef ArchExt) { for (const auto &A : AArch64ARCHExtNames) { if (ArchExt == A.Name) return static_cast(A.ID); } return AArch64::AEK_INVALID; } AArch64::ArchKind AArch64::parseCPUArch(StringRef CPU) { // Resolve aliases first. for (const auto &Alias : AArch64CPUAliases) { if (CPU == Alias.Alias) { CPU = Alias.Name; break; } } // Then find the CPU name. for (const auto &C : AArch64CPUNames) if (CPU == C.Name) return C.ArchID; return ArchKind::INVALID; }