1bdd1243dSDimitry Andric //===-- RISCVTargetParser.cpp - Parser for target features ------*- C++ -*-===// 2bdd1243dSDimitry Andric // 3bdd1243dSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4bdd1243dSDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5bdd1243dSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6bdd1243dSDimitry Andric // 7bdd1243dSDimitry Andric //===----------------------------------------------------------------------===// 8bdd1243dSDimitry Andric // 9bdd1243dSDimitry Andric // This file implements a target parser to recognise hardware features 1006c3fb27SDimitry Andric // for RISC-V CPUs. 11bdd1243dSDimitry Andric // 12bdd1243dSDimitry Andric //===----------------------------------------------------------------------===// 13bdd1243dSDimitry Andric 14bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParser.h" 15bdd1243dSDimitry Andric #include "llvm/ADT/SmallVector.h" 16bdd1243dSDimitry Andric #include "llvm/ADT/StringSwitch.h" 171ac55f4cSDimitry Andric #include "llvm/TargetParser/Triple.h" 18bdd1243dSDimitry Andric 19bdd1243dSDimitry Andric namespace llvm { 20bdd1243dSDimitry Andric namespace RISCV { 21bdd1243dSDimitry Andric 2206c3fb27SDimitry Andric enum CPUKind : unsigned { 23*5f757f3fSDimitry Andric #define PROC(ENUM, NAME, DEFAULT_MARCH, FAST_UNALIGN) CK_##ENUM, 2406c3fb27SDimitry Andric #define TUNE_PROC(ENUM, NAME) CK_##ENUM, 2506c3fb27SDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 2606c3fb27SDimitry Andric }; 2706c3fb27SDimitry Andric 28bdd1243dSDimitry Andric struct CPUInfo { 29bdd1243dSDimitry Andric StringLiteral Name; 30bdd1243dSDimitry Andric StringLiteral DefaultMarch; 31*5f757f3fSDimitry Andric bool FastUnalignedAccess; 32bdd1243dSDimitry Andric bool is64Bit() const { return DefaultMarch.starts_with("rv64"); } 33bdd1243dSDimitry Andric }; 34bdd1243dSDimitry Andric 35bdd1243dSDimitry Andric constexpr CPUInfo RISCVCPUInfo[] = { 36*5f757f3fSDimitry Andric #define PROC(ENUM, NAME, DEFAULT_MARCH, FAST_UNALIGN) \ 37*5f757f3fSDimitry Andric {NAME, DEFAULT_MARCH, FAST_UNALIGN}, 38bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 39bdd1243dSDimitry Andric }; 40bdd1243dSDimitry Andric 4106c3fb27SDimitry Andric static const CPUInfo *getCPUInfoByName(StringRef CPU) { 4206c3fb27SDimitry Andric for (auto &C : RISCVCPUInfo) 4306c3fb27SDimitry Andric if (C.Name == CPU) 4406c3fb27SDimitry Andric return &C; 4506c3fb27SDimitry Andric return nullptr; 4606c3fb27SDimitry Andric } 4706c3fb27SDimitry Andric 48*5f757f3fSDimitry Andric bool hasFastUnalignedAccess(StringRef CPU) { 49*5f757f3fSDimitry Andric const CPUInfo *Info = getCPUInfoByName(CPU); 50*5f757f3fSDimitry Andric return Info && Info->FastUnalignedAccess; 51*5f757f3fSDimitry Andric } 52*5f757f3fSDimitry Andric 5306c3fb27SDimitry Andric bool parseCPU(StringRef CPU, bool IsRV64) { 5406c3fb27SDimitry Andric const CPUInfo *Info = getCPUInfoByName(CPU); 5506c3fb27SDimitry Andric 5606c3fb27SDimitry Andric if (!Info) 57bdd1243dSDimitry Andric return false; 5806c3fb27SDimitry Andric return Info->is64Bit() == IsRV64; 59bdd1243dSDimitry Andric } 60bdd1243dSDimitry Andric 6106c3fb27SDimitry Andric bool parseTuneCPU(StringRef TuneCPU, bool IsRV64) { 6206c3fb27SDimitry Andric std::optional<CPUKind> Kind = 6306c3fb27SDimitry Andric llvm::StringSwitch<std::optional<CPUKind>>(TuneCPU) 64bdd1243dSDimitry Andric #define TUNE_PROC(ENUM, NAME) .Case(NAME, CK_##ENUM) 65bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 6606c3fb27SDimitry Andric .Default(std::nullopt); 6706c3fb27SDimitry Andric 6806c3fb27SDimitry Andric if (Kind.has_value()) 6906c3fb27SDimitry Andric return true; 7006c3fb27SDimitry Andric 7106c3fb27SDimitry Andric // Fallback to parsing as a CPU. 7206c3fb27SDimitry Andric return parseCPU(TuneCPU, IsRV64); 73bdd1243dSDimitry Andric } 74bdd1243dSDimitry Andric 75bdd1243dSDimitry Andric StringRef getMArchFromMcpu(StringRef CPU) { 7606c3fb27SDimitry Andric const CPUInfo *Info = getCPUInfoByName(CPU); 7706c3fb27SDimitry Andric if (!Info) 7806c3fb27SDimitry Andric return ""; 7906c3fb27SDimitry Andric return Info->DefaultMarch; 80bdd1243dSDimitry Andric } 81bdd1243dSDimitry Andric 82bdd1243dSDimitry Andric void fillValidCPUArchList(SmallVectorImpl<StringRef> &Values, bool IsRV64) { 83bdd1243dSDimitry Andric for (const auto &C : RISCVCPUInfo) { 8406c3fb27SDimitry Andric if (IsRV64 == C.is64Bit()) 85bdd1243dSDimitry Andric Values.emplace_back(C.Name); 86bdd1243dSDimitry Andric } 87bdd1243dSDimitry Andric } 88bdd1243dSDimitry Andric 89bdd1243dSDimitry Andric void fillValidTuneCPUArchList(SmallVectorImpl<StringRef> &Values, bool IsRV64) { 90bdd1243dSDimitry Andric for (const auto &C : RISCVCPUInfo) { 9106c3fb27SDimitry Andric if (IsRV64 == C.is64Bit()) 92bdd1243dSDimitry Andric Values.emplace_back(C.Name); 93bdd1243dSDimitry Andric } 94bdd1243dSDimitry Andric #define TUNE_PROC(ENUM, NAME) Values.emplace_back(StringRef(NAME)); 95bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 96bdd1243dSDimitry Andric } 97bdd1243dSDimitry Andric 98bdd1243dSDimitry Andric } // namespace RISCV 99bdd1243dSDimitry Andric } // namespace llvm 100