mirror of
https://github.com/RPCSX/llvm.git
synced 2025-01-19 10:53:55 +00:00
Change how tblgen generates attributes for intrinsics to use a single
switch. With this newfound organization, teach tblgen how not to give all intrinsics the 'nounwind' attribute. Introduce a new intrinsic, llvm.eh.resume, which does not have this attribute. Documentation and uses to follow. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@132252 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
f46337004a
commit
bd0fa4c00d
@ -47,6 +47,9 @@ def IntrReadWriteArgMem : IntrinsicProperty;
|
||||
// Commutative - This intrinsic is commutative: X op Y == Y op X.
|
||||
def Commutative : IntrinsicProperty;
|
||||
|
||||
// Throws - This intrinsic can throw.
|
||||
def Throws : IntrinsicProperty;
|
||||
|
||||
// NoCapture - The specified argument pointer is not captured by the intrinsic.
|
||||
class NoCapture<int argNo> : IntrinsicProperty {
|
||||
int ArgNo = argNo;
|
||||
@ -292,6 +295,7 @@ let Properties = [IntrNoMem] in {
|
||||
def int_eh_exception : Intrinsic<[llvm_ptr_ty], [], [IntrReadMem]>;
|
||||
def int_eh_selector : Intrinsic<[llvm_i32_ty],
|
||||
[llvm_ptr_ty, llvm_ptr_ty, llvm_vararg_ty]>;
|
||||
def int_eh_resume : Intrinsic<[], [llvm_ptr_ty, llvm_i32_ty], [Throws]>;
|
||||
|
||||
def int_eh_typeid_for : Intrinsic<[llvm_i32_ty], [llvm_ptr_ty]>;
|
||||
|
||||
|
@ -69,6 +69,9 @@ namespace llvm {
|
||||
|
||||
/// isCommutative - True if the intrinsic is commutative.
|
||||
bool isCommutative;
|
||||
|
||||
/// canThrow - True if the intrinsic can throw.
|
||||
bool canThrow;
|
||||
|
||||
enum ArgAttribute {
|
||||
NoCapture
|
||||
|
@ -460,6 +460,7 @@ CodeGenIntrinsic::CodeGenIntrinsic(Record *R) {
|
||||
ModRef = ReadWriteMem;
|
||||
isOverloaded = false;
|
||||
isCommutative = false;
|
||||
canThrow = false;
|
||||
|
||||
if (DefName.size() <= 4 ||
|
||||
std::string(DefName.begin(), DefName.begin() + 4) != "int_")
|
||||
@ -582,10 +583,15 @@ CodeGenIntrinsic::CodeGenIntrinsic(Record *R) {
|
||||
ModRef = ReadWriteArgMem;
|
||||
else if (Property->getName() == "Commutative")
|
||||
isCommutative = true;
|
||||
else if (Property->getName() == "Throws")
|
||||
canThrow = true;
|
||||
else if (Property->isSubClassOf("NoCapture")) {
|
||||
unsigned ArgNo = Property->getValueAsInt("ArgNo");
|
||||
ArgumentAttributes.push_back(std::make_pair(ArgNo, NoCapture));
|
||||
} else
|
||||
assert(0 && "Unknown property!");
|
||||
}
|
||||
|
||||
// Sort the argument attributes for later benefit.
|
||||
std::sort(ArgumentAttributes.begin(), ArgumentAttributes.end());
|
||||
}
|
||||
|
@ -465,6 +465,46 @@ void IntrinsicEmitter::EmitGenerator(const std::vector<CodeGenIntrinsic> &Ints,
|
||||
OS << "#endif\n\n";
|
||||
}
|
||||
|
||||
namespace {
|
||||
enum ModRefKind {
|
||||
MRK_none,
|
||||
MRK_readonly,
|
||||
MRK_readnone
|
||||
};
|
||||
|
||||
ModRefKind getModRefKind(const CodeGenIntrinsic &intrinsic) {
|
||||
switch (intrinsic.ModRef) {
|
||||
case CodeGenIntrinsic::NoMem:
|
||||
return MRK_readnone;
|
||||
case CodeGenIntrinsic::ReadArgMem:
|
||||
case CodeGenIntrinsic::ReadMem:
|
||||
return MRK_readonly;
|
||||
case CodeGenIntrinsic::ReadWriteArgMem:
|
||||
case CodeGenIntrinsic::ReadWriteMem:
|
||||
return MRK_none;
|
||||
}
|
||||
assert(0 && "bad mod-ref kind");
|
||||
return MRK_none;
|
||||
}
|
||||
|
||||
struct AttributeComparator {
|
||||
bool operator()(const CodeGenIntrinsic *L, const CodeGenIntrinsic *R) const {
|
||||
// Sort throwing intrinsics after non-throwing intrinsics.
|
||||
if (L->canThrow != R->canThrow)
|
||||
return R->canThrow;
|
||||
|
||||
// Try to order by readonly/readnone attribute.
|
||||
ModRefKind LK = getModRefKind(*L);
|
||||
ModRefKind RK = getModRefKind(*R);
|
||||
if (LK != RK) return (LK > RK);
|
||||
|
||||
// Order by argument attributes.
|
||||
// This is reliable because each side is already sorted internally.
|
||||
return (L->ArgumentAttributes < R->ArgumentAttributes);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
/// EmitAttributes - This emits the Intrinsic::getAttributes method.
|
||||
void IntrinsicEmitter::
|
||||
EmitAttributes(const std::vector<CodeGenIntrinsic> &Ints, raw_ostream &OS) {
|
||||
@ -472,84 +512,96 @@ EmitAttributes(const std::vector<CodeGenIntrinsic> &Ints, raw_ostream &OS) {
|
||||
OS << "#ifdef GET_INTRINSIC_ATTRIBUTES\n";
|
||||
if (TargetOnly)
|
||||
OS << "static AttrListPtr getAttributes(" << TargetPrefix
|
||||
<< "Intrinsic::ID id) {";
|
||||
<< "Intrinsic::ID id) {\n";
|
||||
else
|
||||
OS << "AttrListPtr Intrinsic::getAttributes(ID id) {";
|
||||
OS << " // No intrinsic can throw exceptions.\n";
|
||||
OS << " Attributes Attr = Attribute::NoUnwind;\n";
|
||||
OS << " switch (id) {\n";
|
||||
OS << " default: break;\n";
|
||||
unsigned MaxArgAttrs = 0;
|
||||
OS << "AttrListPtr Intrinsic::getAttributes(ID id) {\n";
|
||||
|
||||
// Compute the maximum number of attribute arguments.
|
||||
std::vector<const CodeGenIntrinsic*> sortedIntrinsics(Ints.size());
|
||||
unsigned maxArgAttrs = 0;
|
||||
for (unsigned i = 0, e = Ints.size(); i != e; ++i) {
|
||||
MaxArgAttrs =
|
||||
std::max(MaxArgAttrs, unsigned(Ints[i].ArgumentAttributes.size()));
|
||||
switch (Ints[i].ModRef) {
|
||||
default: break;
|
||||
case CodeGenIntrinsic::NoMem:
|
||||
OS << " case " << TargetPrefix << "Intrinsic::" << Ints[i].EnumName
|
||||
<< ":\n";
|
||||
break;
|
||||
}
|
||||
const CodeGenIntrinsic &intrinsic = Ints[i];
|
||||
sortedIntrinsics[i] = &intrinsic;
|
||||
maxArgAttrs =
|
||||
std::max(maxArgAttrs, unsigned(intrinsic.ArgumentAttributes.size()));
|
||||
}
|
||||
OS << " Attr |= Attribute::ReadNone; // These do not access memory.\n";
|
||||
OS << " break;\n";
|
||||
for (unsigned i = 0, e = Ints.size(); i != e; ++i) {
|
||||
switch (Ints[i].ModRef) {
|
||||
default: break;
|
||||
case CodeGenIntrinsic::ReadArgMem:
|
||||
case CodeGenIntrinsic::ReadMem:
|
||||
OS << " case " << TargetPrefix << "Intrinsic::" << Ints[i].EnumName
|
||||
<< ":\n";
|
||||
break;
|
||||
}
|
||||
}
|
||||
OS << " Attr |= Attribute::ReadOnly; // These do not write memory.\n";
|
||||
OS << " break;\n";
|
||||
OS << " }\n";
|
||||
OS << " AttributeWithIndex AWI[" << MaxArgAttrs+1 << "];\n";
|
||||
|
||||
// Emit an array of AttributeWithIndex. Most intrinsics will have
|
||||
// at least one entry, for the function itself (index ~1), which is
|
||||
// usually nounwind.
|
||||
OS << " AttributeWithIndex AWI[" << maxArgAttrs+1 << "];\n";
|
||||
OS << " unsigned NumAttrs = 0;\n";
|
||||
OS << " switch (id) {\n";
|
||||
OS << " default: break;\n";
|
||||
|
||||
// Add argument attributes for any intrinsics that have them.
|
||||
for (unsigned i = 0, e = Ints.size(); i != e; ++i) {
|
||||
if (Ints[i].ArgumentAttributes.empty()) continue;
|
||||
|
||||
OS << " case " << TargetPrefix << "Intrinsic::" << Ints[i].EnumName
|
||||
<< ":\n";
|
||||
OS << " default: break;\n";
|
||||
|
||||
std::vector<std::pair<unsigned, CodeGenIntrinsic::ArgAttribute> > ArgAttrs =
|
||||
Ints[i].ArgumentAttributes;
|
||||
// Sort by argument index.
|
||||
std::sort(ArgAttrs.begin(), ArgAttrs.end());
|
||||
AttributeComparator precedes;
|
||||
|
||||
unsigned NumArgsWithAttrs = 0;
|
||||
std::stable_sort(sortedIntrinsics.begin(), sortedIntrinsics.end(), precedes);
|
||||
|
||||
while (!ArgAttrs.empty()) {
|
||||
unsigned ArgNo = ArgAttrs[0].first;
|
||||
for (unsigned i = 0, e = sortedIntrinsics.size(); i != e; ++i) {
|
||||
const CodeGenIntrinsic &intrinsic = *sortedIntrinsics[i];
|
||||
OS << " case " << TargetPrefix << "Intrinsic::"
|
||||
<< intrinsic.EnumName << ":\n";
|
||||
|
||||
// Fill out the case if this is the last case for this range of
|
||||
// intrinsics.
|
||||
if (i + 1 != e && !precedes(&intrinsic, sortedIntrinsics[i + 1]))
|
||||
continue;
|
||||
|
||||
// Keep track of the number of attributes we're writing out.
|
||||
unsigned numAttrs = 0;
|
||||
|
||||
// The argument attributes are alreadys sorted by argument index.
|
||||
for (unsigned ai = 0, ae = intrinsic.ArgumentAttributes.size(); ai != ae;) {
|
||||
unsigned argNo = intrinsic.ArgumentAttributes[ai].first;
|
||||
|
||||
OS << " AWI[" << NumArgsWithAttrs++ << "] = AttributeWithIndex::get("
|
||||
<< ArgNo+1 << ", 0";
|
||||
OS << " AWI[" << numAttrs++ << "] = AttributeWithIndex::get("
|
||||
<< argNo+1 << ", ";
|
||||
|
||||
while (!ArgAttrs.empty() && ArgAttrs[0].first == ArgNo) {
|
||||
switch (ArgAttrs[0].second) {
|
||||
default: assert(0 && "Unknown arg attribute");
|
||||
bool moreThanOne = false;
|
||||
|
||||
do {
|
||||
if (moreThanOne) OS << '|';
|
||||
|
||||
switch (intrinsic.ArgumentAttributes[ai].second) {
|
||||
case CodeGenIntrinsic::NoCapture:
|
||||
OS << "|Attribute::NoCapture";
|
||||
OS << "Attribute::NoCapture";
|
||||
break;
|
||||
}
|
||||
ArgAttrs.erase(ArgAttrs.begin());
|
||||
|
||||
++ai;
|
||||
moreThanOne = true;
|
||||
} while (ai != ae && intrinsic.ArgumentAttributes[ai].first == argNo);
|
||||
|
||||
OS << ");\n";
|
||||
}
|
||||
|
||||
ModRefKind modRef = getModRefKind(intrinsic);
|
||||
|
||||
if (!intrinsic.canThrow || modRef) {
|
||||
OS << " AWI[" << numAttrs++ << "] = AttributeWithIndex::get(~0, ";
|
||||
if (!intrinsic.canThrow) {
|
||||
OS << "Attribute::NoUnwind";
|
||||
if (modRef) OS << '|';
|
||||
}
|
||||
switch (modRef) {
|
||||
case MRK_none: break;
|
||||
case MRK_readonly: OS << "Attribute::ReadOnly"; break;
|
||||
case MRK_readnone: OS << "Attribute::ReadNone"; break;
|
||||
}
|
||||
OS << ");\n";
|
||||
}
|
||||
|
||||
OS << " NumAttrs = " << NumArgsWithAttrs << ";\n";
|
||||
OS << " break;\n";
|
||||
|
||||
if (numAttrs) {
|
||||
OS << " NumAttrs = " << numAttrs << ";\n";
|
||||
OS << " break;\n";
|
||||
} else {
|
||||
OS << " return AttrListPtr();\n";
|
||||
}
|
||||
}
|
||||
|
||||
OS << " }\n";
|
||||
OS << " AWI[NumAttrs] = AttributeWithIndex::get(~0, Attr);\n";
|
||||
OS << " return AttrListPtr::get(AWI, NumAttrs+1);\n";
|
||||
OS << " return AttrListPtr::get(AWI, NumAttrs);\n";
|
||||
OS << "}\n";
|
||||
OS << "#endif // GET_INTRINSIC_ATTRIBUTES\n\n";
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user