10b57cec5SDimitry Andric //===--- BPF.cpp - Implement BPF target feature support -------------------===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric // 90b57cec5SDimitry Andric // This file implements BPF TargetInfo objects. 100b57cec5SDimitry Andric // 110b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 120b57cec5SDimitry Andric 130b57cec5SDimitry Andric #include "BPF.h" 140b57cec5SDimitry Andric #include "Targets.h" 150b57cec5SDimitry Andric #include "clang/Basic/MacroBuilder.h" 16a7dea167SDimitry Andric #include "clang/Basic/TargetBuiltins.h" 170b57cec5SDimitry Andric #include "llvm/ADT/StringRef.h" 180b57cec5SDimitry Andric 190b57cec5SDimitry Andric using namespace clang; 200b57cec5SDimitry Andric using namespace clang::targets; 210b57cec5SDimitry Andric 22bdd1243dSDimitry Andric static constexpr Builtin::Info BuiltinInfo[] = { 23a7dea167SDimitry Andric #define BUILTIN(ID, TYPE, ATTRS) \ 24bdd1243dSDimitry Andric {#ID, TYPE, ATTRS, nullptr, HeaderDesc::NO_HEADER, ALL_LANGUAGES}, 25a7dea167SDimitry Andric #include "clang/Basic/BuiltinsBPF.def" 26a7dea167SDimitry Andric }; 27a7dea167SDimitry Andric 280b57cec5SDimitry Andric void BPFTargetInfo::getTargetDefines(const LangOptions &Opts, 290b57cec5SDimitry Andric MacroBuilder &Builder) const { 300b57cec5SDimitry Andric Builder.defineMacro("__bpf__"); 310b57cec5SDimitry Andric Builder.defineMacro("__BPF__"); 32*5f757f3fSDimitry Andric 33*5f757f3fSDimitry Andric std::string CPU = getTargetOpts().CPU; 34*5f757f3fSDimitry Andric if (CPU == "probe") { 35*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_CPU_VERSION__", "0"); 36*5f757f3fSDimitry Andric return; 37*5f757f3fSDimitry Andric } 38*5f757f3fSDimitry Andric if (CPU.empty() || CPU == "generic" || CPU == "v1") { 39*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_CPU_VERSION__", "1"); 40*5f757f3fSDimitry Andric return; 41*5f757f3fSDimitry Andric } 42*5f757f3fSDimitry Andric 43*5f757f3fSDimitry Andric std::string CpuVerNumStr = CPU.substr(1); 44*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_CPU_VERSION__", CpuVerNumStr); 45*5f757f3fSDimitry Andric 46*5f757f3fSDimitry Andric int CpuVerNum = std::stoi(CpuVerNumStr); 47*5f757f3fSDimitry Andric if (CpuVerNum >= 2) 48*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_JMP_EXT"); 49*5f757f3fSDimitry Andric 50*5f757f3fSDimitry Andric if (CpuVerNum >= 3) { 51*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_JMP32"); 52*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_ALU32"); 53*5f757f3fSDimitry Andric } 54*5f757f3fSDimitry Andric 55*5f757f3fSDimitry Andric if (CpuVerNum >= 4) { 56*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_LDSX"); 57*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_MOVSX"); 58*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_BSWAP"); 59*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_SDIV_SMOD"); 60*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_GOTOL"); 61*5f757f3fSDimitry Andric Builder.defineMacro("__BPF_FEATURE_ST"); 62*5f757f3fSDimitry Andric } 630b57cec5SDimitry Andric } 640b57cec5SDimitry Andric 650b57cec5SDimitry Andric static constexpr llvm::StringLiteral ValidCPUNames[] = {"generic", "v1", "v2", 66*5f757f3fSDimitry Andric "v3", "v4", "probe"}; 670b57cec5SDimitry Andric 680b57cec5SDimitry Andric bool BPFTargetInfo::isValidCPUName(StringRef Name) const { 69349cc55cSDimitry Andric return llvm::is_contained(ValidCPUNames, Name); 700b57cec5SDimitry Andric } 710b57cec5SDimitry Andric 720b57cec5SDimitry Andric void BPFTargetInfo::fillValidCPUList(SmallVectorImpl<StringRef> &Values) const { 730b57cec5SDimitry Andric Values.append(std::begin(ValidCPUNames), std::end(ValidCPUNames)); 740b57cec5SDimitry Andric } 75a7dea167SDimitry Andric 76a7dea167SDimitry Andric ArrayRef<Builtin::Info> BPFTargetInfo::getTargetBuiltins() const { 77bdd1243dSDimitry Andric return llvm::ArrayRef(BuiltinInfo, 78bdd1243dSDimitry Andric clang::BPF::LastTSBuiltin - Builtin::FirstTSBuiltin); 79a7dea167SDimitry Andric } 80fe6060f1SDimitry Andric 81fe6060f1SDimitry Andric bool BPFTargetInfo::handleTargetFeatures(std::vector<std::string> &Features, 82fe6060f1SDimitry Andric DiagnosticsEngine &Diags) { 83fe6060f1SDimitry Andric for (const auto &Feature : Features) { 84fe6060f1SDimitry Andric if (Feature == "+alu32") { 85fe6060f1SDimitry Andric HasAlu32 = true; 86fe6060f1SDimitry Andric } 87fe6060f1SDimitry Andric } 88fe6060f1SDimitry Andric 89fe6060f1SDimitry Andric return true; 90fe6060f1SDimitry Andric } 91