xref: /freebsd/contrib/llvm-project/clang/lib/Basic/Targets/BPF.cpp (revision 5f757f3ff9144b609b3c433dfd370cc6bdc191ad)
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