2015-11-11 21:35:42 +01:00
|
|
|
//===- Attributes.cpp - Generate attributes -------------------------------===//
|
|
|
|
//
|
2019-01-19 09:50:56 +01:00
|
|
|
// 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
|
2015-11-11 21:35:42 +01:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include "llvm/TableGen/Record.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "attr-enum"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class Attributes {
|
|
|
|
public:
|
|
|
|
Attributes(RecordKeeper &R) : Records(R) {}
|
|
|
|
void emit(raw_ostream &OS);
|
|
|
|
|
|
|
|
private:
|
2020-02-02 14:46:50 +01:00
|
|
|
void emitTargetIndependentNames(raw_ostream &OS);
|
2015-12-23 00:57:37 +01:00
|
|
|
void emitFnAttrCompatCheck(raw_ostream &OS, bool IsStringAttr);
|
2021-07-11 16:54:03 +02:00
|
|
|
void emitAttributeProperties(raw_ostream &OF);
|
2015-12-23 00:57:37 +01:00
|
|
|
|
2015-11-11 21:35:42 +01:00
|
|
|
RecordKeeper &Records;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // End anonymous namespace.
|
|
|
|
|
2020-02-02 14:46:50 +01:00
|
|
|
void Attributes::emitTargetIndependentNames(raw_ostream &OS) {
|
|
|
|
OS << "#ifdef GET_ATTR_NAMES\n";
|
|
|
|
OS << "#undef GET_ATTR_NAMES\n";
|
2015-11-11 21:35:42 +01:00
|
|
|
|
2020-02-02 14:46:50 +01:00
|
|
|
OS << "#ifndef ATTRIBUTE_ALL\n";
|
|
|
|
OS << "#define ATTRIBUTE_ALL(FIRST, SECOND)\n";
|
|
|
|
OS << "#endif\n\n";
|
2015-11-11 21:35:42 +01:00
|
|
|
|
2020-04-26 16:52:53 +02:00
|
|
|
auto Emit = [&](ArrayRef<StringRef> KindNames, StringRef MacroName) {
|
2020-02-02 14:46:50 +01:00
|
|
|
OS << "#ifndef " << MacroName << "\n";
|
2020-04-26 16:52:53 +02:00
|
|
|
OS << "#define " << MacroName
|
|
|
|
<< "(FIRST, SECOND) ATTRIBUTE_ALL(FIRST, SECOND)\n";
|
2020-02-02 14:46:50 +01:00
|
|
|
OS << "#endif\n\n";
|
2020-04-26 16:52:53 +02:00
|
|
|
for (StringRef KindName : KindNames) {
|
|
|
|
for (auto A : Records.getAllDerivedDefinitions(KindName)) {
|
|
|
|
OS << MacroName << "(" << A->getName() << ","
|
|
|
|
<< A->getValueAsString("AttrString") << ")\n";
|
|
|
|
}
|
2020-02-02 14:46:50 +01:00
|
|
|
}
|
|
|
|
OS << "#undef " << MacroName << "\n\n";
|
|
|
|
};
|
2016-04-20 03:02:12 +02:00
|
|
|
|
2020-04-26 16:52:53 +02:00
|
|
|
// Emit attribute enums in the same order llvm::Attribute::operator< expects.
|
|
|
|
Emit({"EnumAttr", "TypeAttr", "IntAttr"}, "ATTRIBUTE_ENUM");
|
|
|
|
Emit({"StrBoolAttr"}, "ATTRIBUTE_STRBOOL");
|
2016-04-20 03:02:12 +02:00
|
|
|
|
2020-02-02 14:46:50 +01:00
|
|
|
OS << "#undef ATTRIBUTE_ALL\n";
|
2021-07-10 18:36:00 +02:00
|
|
|
OS << "#endif\n\n";
|
|
|
|
|
|
|
|
OS << "#ifdef GET_ATTR_ENUM\n";
|
|
|
|
OS << "#undef GET_ATTR_ENUM\n";
|
|
|
|
unsigned Value = 1; // Leave zero for AttrKind::None.
|
|
|
|
for (StringRef KindName : {"EnumAttr", "TypeAttr", "IntAttr"}) {
|
|
|
|
OS << "First" << KindName << " = " << Value << ",\n";
|
|
|
|
for (auto A : Records.getAllDerivedDefinitions(KindName)) {
|
|
|
|
OS << A->getName() << " = " << Value << ",\n";
|
|
|
|
Value++;
|
|
|
|
}
|
|
|
|
OS << "Last" << KindName << " = " << (Value - 1) << ",\n";
|
|
|
|
}
|
|
|
|
OS << "#endif\n\n";
|
2016-04-20 03:02:12 +02:00
|
|
|
}
|
|
|
|
|
2015-12-23 00:57:37 +01:00
|
|
|
void Attributes::emitFnAttrCompatCheck(raw_ostream &OS, bool IsStringAttr) {
|
|
|
|
OS << "#ifdef GET_ATTR_COMPAT_FUNC\n";
|
|
|
|
OS << "#undef GET_ATTR_COMPAT_FUNC\n";
|
|
|
|
|
|
|
|
OS << "static inline bool hasCompatibleFnAttrs(const Function &Caller,\n"
|
|
|
|
<< " const Function &Callee) {\n";
|
|
|
|
OS << " bool Ret = true;\n\n";
|
|
|
|
|
|
|
|
std::vector<Record *> CompatRules =
|
|
|
|
Records.getAllDerivedDefinitions("CompatRule");
|
|
|
|
|
|
|
|
for (auto *Rule : CompatRules) {
|
2017-05-31 23:12:46 +02:00
|
|
|
StringRef FuncName = Rule->getValueAsString("CompatFunc");
|
2015-12-23 00:57:37 +01:00
|
|
|
OS << " Ret &= " << FuncName << "(Caller, Callee);\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
OS << "\n";
|
|
|
|
OS << " return Ret;\n";
|
|
|
|
OS << "}\n\n";
|
|
|
|
|
|
|
|
std::vector<Record *> MergeRules =
|
|
|
|
Records.getAllDerivedDefinitions("MergeRule");
|
|
|
|
OS << "static inline void mergeFnAttrs(Function &Caller,\n"
|
|
|
|
<< " const Function &Callee) {\n";
|
|
|
|
|
|
|
|
for (auto *Rule : MergeRules) {
|
2017-05-31 23:12:46 +02:00
|
|
|
StringRef FuncName = Rule->getValueAsString("MergeFunc");
|
2015-12-23 00:57:37 +01:00
|
|
|
OS << " " << FuncName << "(Caller, Callee);\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
OS << "}\n\n";
|
|
|
|
|
|
|
|
OS << "#endif\n";
|
|
|
|
}
|
|
|
|
|
2021-07-11 16:54:03 +02:00
|
|
|
void Attributes::emitAttributeProperties(raw_ostream &OS) {
|
|
|
|
OS << "#ifdef GET_ATTR_PROP_TABLE\n";
|
|
|
|
OS << "#undef GET_ATTR_PROP_TABLE\n";
|
|
|
|
OS << "static const uint8_t AttrPropTable[] = {\n";
|
|
|
|
for (StringRef KindName : {"EnumAttr", "TypeAttr", "IntAttr"}) {
|
|
|
|
for (auto A : Records.getAllDerivedDefinitions(KindName)) {
|
|
|
|
OS << "0";
|
|
|
|
for (Init *P : *A->getValueAsListInit("Properties"))
|
|
|
|
OS << " | AttributeProperty::" << cast<DefInit>(P)->getDef()->getName();
|
|
|
|
OS << ",\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
OS << "};\n";
|
|
|
|
OS << "#endif\n";
|
|
|
|
}
|
|
|
|
|
2015-11-11 21:35:42 +01:00
|
|
|
void Attributes::emit(raw_ostream &OS) {
|
2020-02-02 14:46:50 +01:00
|
|
|
emitTargetIndependentNames(OS);
|
2015-12-23 00:57:37 +01:00
|
|
|
emitFnAttrCompatCheck(OS, false);
|
2021-07-11 16:54:03 +02:00
|
|
|
emitAttributeProperties(OS);
|
2015-11-11 21:35:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
void EmitAttributes(RecordKeeper &RK, raw_ostream &OS) {
|
|
|
|
Attributes(RK).emit(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End llvm namespace.
|