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 10*06c3fb27SDimitry 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 22*06c3fb27SDimitry Andric enum CPUKind : unsigned { 23*06c3fb27SDimitry Andric #define PROC(ENUM, NAME, DEFAULT_MARCH) CK_##ENUM, 24*06c3fb27SDimitry Andric #define TUNE_PROC(ENUM, NAME) CK_##ENUM, 25*06c3fb27SDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 26*06c3fb27SDimitry Andric }; 27*06c3fb27SDimitry Andric 28bdd1243dSDimitry Andric struct CPUInfo { 29bdd1243dSDimitry Andric StringLiteral Name; 30bdd1243dSDimitry Andric StringLiteral DefaultMarch; 31bdd1243dSDimitry Andric bool is64Bit() const { return DefaultMarch.starts_with("rv64"); } 32bdd1243dSDimitry Andric }; 33bdd1243dSDimitry Andric 34bdd1243dSDimitry Andric constexpr CPUInfo RISCVCPUInfo[] = { 35bdd1243dSDimitry Andric #define PROC(ENUM, NAME, DEFAULT_MARCH) \ 36*06c3fb27SDimitry Andric {NAME, DEFAULT_MARCH}, 37bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 38bdd1243dSDimitry Andric }; 39bdd1243dSDimitry Andric 40*06c3fb27SDimitry Andric static const CPUInfo *getCPUInfoByName(StringRef CPU) { 41*06c3fb27SDimitry Andric for (auto &C : RISCVCPUInfo) 42*06c3fb27SDimitry Andric if (C.Name == CPU) 43*06c3fb27SDimitry Andric return &C; 44*06c3fb27SDimitry Andric return nullptr; 45*06c3fb27SDimitry Andric } 46*06c3fb27SDimitry Andric 47*06c3fb27SDimitry Andric bool parseCPU(StringRef CPU, bool IsRV64) { 48*06c3fb27SDimitry Andric const CPUInfo *Info = getCPUInfoByName(CPU); 49*06c3fb27SDimitry Andric 50*06c3fb27SDimitry Andric if (!Info) 51bdd1243dSDimitry Andric return false; 52*06c3fb27SDimitry Andric return Info->is64Bit() == IsRV64; 53bdd1243dSDimitry Andric } 54bdd1243dSDimitry Andric 55*06c3fb27SDimitry Andric bool parseTuneCPU(StringRef TuneCPU, bool IsRV64) { 56*06c3fb27SDimitry Andric std::optional<CPUKind> Kind = 57*06c3fb27SDimitry Andric llvm::StringSwitch<std::optional<CPUKind>>(TuneCPU) 58bdd1243dSDimitry Andric #define TUNE_PROC(ENUM, NAME) .Case(NAME, CK_##ENUM) 59bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 60*06c3fb27SDimitry Andric .Default(std::nullopt); 61*06c3fb27SDimitry Andric 62*06c3fb27SDimitry Andric if (Kind.has_value()) 63*06c3fb27SDimitry Andric return true; 64*06c3fb27SDimitry Andric 65*06c3fb27SDimitry Andric // Fallback to parsing as a CPU. 66*06c3fb27SDimitry Andric return parseCPU(TuneCPU, IsRV64); 67bdd1243dSDimitry Andric } 68bdd1243dSDimitry Andric 69bdd1243dSDimitry Andric StringRef getMArchFromMcpu(StringRef CPU) { 70*06c3fb27SDimitry Andric const CPUInfo *Info = getCPUInfoByName(CPU); 71*06c3fb27SDimitry Andric if (!Info) 72*06c3fb27SDimitry Andric return ""; 73*06c3fb27SDimitry Andric return Info->DefaultMarch; 74bdd1243dSDimitry Andric } 75bdd1243dSDimitry Andric 76bdd1243dSDimitry Andric void fillValidCPUArchList(SmallVectorImpl<StringRef> &Values, bool IsRV64) { 77bdd1243dSDimitry Andric for (const auto &C : RISCVCPUInfo) { 78*06c3fb27SDimitry Andric if (IsRV64 == C.is64Bit()) 79bdd1243dSDimitry Andric Values.emplace_back(C.Name); 80bdd1243dSDimitry Andric } 81bdd1243dSDimitry Andric } 82bdd1243dSDimitry Andric 83bdd1243dSDimitry Andric void fillValidTuneCPUArchList(SmallVectorImpl<StringRef> &Values, bool IsRV64) { 84bdd1243dSDimitry Andric for (const auto &C : RISCVCPUInfo) { 85*06c3fb27SDimitry Andric if (IsRV64 == C.is64Bit()) 86bdd1243dSDimitry Andric Values.emplace_back(C.Name); 87bdd1243dSDimitry Andric } 88bdd1243dSDimitry Andric #define TUNE_PROC(ENUM, NAME) Values.emplace_back(StringRef(NAME)); 89bdd1243dSDimitry Andric #include "llvm/TargetParser/RISCVTargetParserDef.inc" 90bdd1243dSDimitry Andric } 91bdd1243dSDimitry Andric 92bdd1243dSDimitry Andric } // namespace RISCV 93bdd1243dSDimitry Andric } // namespace llvm 94