mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-13 04:30:23 +00:00
Improve support for type-generic vector intrinsics by teaching TableGen how
to handle LLVMMatchType intrinsic parameters, and by adding new subclasses of LLVMMatchType to match vector types with integral elements that are either twice as wide or half as wide as the elements of the matched type. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@61834 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
0a79a2f8b0
commit
bc03979536
@ -369,6 +369,26 @@ public:
|
||||
return VectorType::get(EltTy, VTy->getNumElements());
|
||||
}
|
||||
|
||||
/// VectorType::getExtendedElementVectorType - This static method is like
|
||||
/// getInteger except that the element types are twice as wide as the
|
||||
/// elements in the input type.
|
||||
///
|
||||
static VectorType *getExtendedElementVectorType(const VectorType *VTy) {
|
||||
unsigned EltBits = VTy->getElementType()->getPrimitiveSizeInBits();
|
||||
const Type *EltTy = IntegerType::get(EltBits * 2);
|
||||
return VectorType::get(EltTy, VTy->getNumElements());
|
||||
}
|
||||
|
||||
/// VectorType::getTruncatedElementVectorType - This static method is like
|
||||
/// getInteger except that the element types are half as wide as the
|
||||
/// elements in the input type.
|
||||
///
|
||||
static VectorType *getTruncatedElementVectorType(const VectorType *VTy) {
|
||||
unsigned EltBits = VTy->getElementType()->getPrimitiveSizeInBits();
|
||||
const Type *EltTy = IntegerType::get(EltBits / 2);
|
||||
return VectorType::get(EltTy, VTy->getNumElements());
|
||||
}
|
||||
|
||||
/// @brief Return the number of elements in the Vector type.
|
||||
inline unsigned getNumElements() const { return NumElements; }
|
||||
|
||||
|
@ -74,6 +74,13 @@ class LLVMMatchType<int num>
|
||||
int Number = num;
|
||||
}
|
||||
|
||||
// Match the type of another intrinsic parameter that is expected to be
|
||||
// an integral vector type, but change the element size to be twice as wide
|
||||
// or half as wide as the other type. This is only useful when the intrinsic
|
||||
// is overloaded, so the matched type should be declared as iAny.
|
||||
class LLVMExtendedElementVectorType<int num> : LLVMMatchType<num>;
|
||||
class LLVMTruncatedElementVectorType<int num> : LLVMMatchType<num>;
|
||||
|
||||
def llvm_void_ty : LLVMType<isVoid>;
|
||||
def llvm_anyint_ty : LLVMType<iAny>;
|
||||
def llvm_anyfloat_ty : LLVMType<fAny>;
|
||||
|
@ -1313,6 +1313,11 @@ void Verifier::visitInstruction(Instruction &I) {
|
||||
InstsInThisBlock.insert(&I);
|
||||
}
|
||||
|
||||
// Flags used by TableGen to mark intrinsic parameters with the
|
||||
// LLVMExtendedElementVectorType and LLVMTruncatedElementVectorType classes.
|
||||
static const unsigned ExtendedElementVectorType = 0x40000000;
|
||||
static const unsigned TruncatedElementVectorType = 0x20000000;
|
||||
|
||||
/// visitIntrinsicFunction - Allow intrinsics to be verified in different ways.
|
||||
///
|
||||
void Verifier::visitIntrinsicFunctionCall(Intrinsic::ID ID, CallInst &CI) {
|
||||
@ -1376,13 +1381,34 @@ bool Verifier::PerformTypeCheck(Intrinsic::ID ID, Function *F, const Type *Ty,
|
||||
|
||||
unsigned NumElts = 0;
|
||||
const Type *EltTy = Ty;
|
||||
if (const VectorType *VTy = dyn_cast<VectorType>(Ty)) {
|
||||
const VectorType *VTy = dyn_cast<VectorType>(Ty);
|
||||
if (VTy) {
|
||||
EltTy = VTy->getElementType();
|
||||
NumElts = VTy->getNumElements();
|
||||
}
|
||||
|
||||
if (VT < 0) {
|
||||
int Match = ~VT;
|
||||
|
||||
// Check flags that indicate a type that is an integral vector type with
|
||||
// elements that are larger or smaller than the elements of the matched
|
||||
// type.
|
||||
if ((Match & (ExtendedElementVectorType |
|
||||
TruncatedElementVectorType)) != 0) {
|
||||
if (!VTy) {
|
||||
CheckFailed("Intrinsic parameter #" + utostr(ArgNo - 1) + " is not "
|
||||
"a vector type.", F);
|
||||
return false;
|
||||
}
|
||||
// Adjust the current Ty (in the opposite direction) rather than
|
||||
// the type being matched against.
|
||||
if ((Match & ExtendedElementVectorType) != 0)
|
||||
Ty = VectorType::getTruncatedElementVectorType(VTy);
|
||||
else
|
||||
Ty = VectorType::getExtendedElementVectorType(VTy);
|
||||
Match &= ~(ExtendedElementVectorType | TruncatedElementVectorType);
|
||||
}
|
||||
|
||||
const Type *RetTy = FTy->getReturnType();
|
||||
const StructType *ST = dyn_cast<StructType>(RetTy);
|
||||
unsigned NumRets = 1;
|
||||
|
@ -483,7 +483,17 @@ CodeGenIntrinsic::CodeGenIntrinsic(Record *R) {
|
||||
for (unsigned i = 0, e = TypeList->getSize(); i != e; ++i) {
|
||||
Record *TyEl = TypeList->getElementAsRecord(i);
|
||||
assert(TyEl->isSubClassOf("LLVMType") && "Expected a type!");
|
||||
MVT::SimpleValueType VT = getValueType(TyEl->getValueAsDef("VT"));
|
||||
MVT::SimpleValueType VT;
|
||||
if (TyEl->isSubClassOf("LLVMMatchType")) {
|
||||
VT = IS.RetVTs[TyEl->getValueAsInt("Number")];
|
||||
// It only makes sense to use the extended and truncated vector element
|
||||
// variants with iAny types; otherwise, if the intrinsic is not
|
||||
// overloaded, all the types can be specified directly.
|
||||
assert(((!TyEl->isSubClassOf("LLVMExtendedElementVectorType") &&
|
||||
!TyEl->isSubClassOf("LLVMTruncatedElementVectorType")) ||
|
||||
VT == MVT::iAny) && "Expected iAny type");
|
||||
} else
|
||||
VT = getValueType(TyEl->getValueAsDef("VT"));
|
||||
isOverloaded |= VT == MVT::iAny || VT == MVT::fAny || VT == MVT::iPTRAny;
|
||||
IS.RetVTs.push_back(VT);
|
||||
IS.RetTypeDefs.push_back(TyEl);
|
||||
@ -497,7 +507,21 @@ CodeGenIntrinsic::CodeGenIntrinsic(Record *R) {
|
||||
for (unsigned i = 0, e = TypeList->getSize(); i != e; ++i) {
|
||||
Record *TyEl = TypeList->getElementAsRecord(i);
|
||||
assert(TyEl->isSubClassOf("LLVMType") && "Expected a type!");
|
||||
MVT::SimpleValueType VT = getValueType(TyEl->getValueAsDef("VT"));
|
||||
MVT::SimpleValueType VT;
|
||||
if (TyEl->isSubClassOf("LLVMMatchType")) {
|
||||
unsigned MatchTy = TyEl->getValueAsInt("Number");
|
||||
if (MatchTy < IS.RetVTs.size())
|
||||
VT = IS.RetVTs[MatchTy];
|
||||
else
|
||||
VT = IS.ParamVTs[MatchTy - IS.RetVTs.size()];
|
||||
// It only makes sense to use the extended and truncated vector element
|
||||
// variants with iAny types; otherwise, if the intrinsic is not
|
||||
// overloaded, all the types can be specified directly.
|
||||
assert(((!TyEl->isSubClassOf("LLVMExtendedElementVectorType") &&
|
||||
!TyEl->isSubClassOf("LLVMTruncatedElementVectorType")) ||
|
||||
VT == MVT::iAny) && "Expected iAny type");
|
||||
} else
|
||||
VT = getValueType(TyEl->getValueAsDef("VT"));
|
||||
isOverloaded |= VT == MVT::iAny || VT == MVT::fAny || VT == MVT::iPTRAny;
|
||||
IS.ParamVTs.push_back(VT);
|
||||
IS.ParamTypeDefs.push_back(TyEl);
|
||||
|
@ -167,7 +167,14 @@ static void EmitTypeGenerate(std::ostream &OS, const Record *ArgType,
|
||||
if (ArgType->isSubClassOf("LLVMMatchType")) {
|
||||
unsigned Number = ArgType->getValueAsInt("Number");
|
||||
assert(Number < ArgNo && "Invalid matching number!");
|
||||
OS << "Tys[" << Number << "]";
|
||||
if (ArgType->isSubClassOf("LLVMExtendedElementVectorType"))
|
||||
OS << "VectorType::getExtendedElementVectorType"
|
||||
<< "(dyn_cast<VectorType>(Tys[" << Number << "]))";
|
||||
else if (ArgType->isSubClassOf("LLVMTruncatedElementVectorType"))
|
||||
OS << "VectorType::getTruncatedElementVectorType"
|
||||
<< "(dyn_cast<VectorType>(Tys[" << Number << "]))";
|
||||
else
|
||||
OS << "Tys[" << Number << "]";
|
||||
} else if (VT == MVT::iAny || VT == MVT::fAny) {
|
||||
// NOTE: The ArgNo variable here is not the absolute argument number, it is
|
||||
// the index of the "arbitrary" type in the Tys array passed to the
|
||||
@ -281,7 +288,12 @@ void IntrinsicEmitter::EmitVerifier(const std::vector<CodeGenIntrinsic> &Ints,
|
||||
if (ArgType->isSubClassOf("LLVMMatchType")) {
|
||||
unsigned Number = ArgType->getValueAsInt("Number");
|
||||
assert(Number < j && "Invalid matching number!");
|
||||
OS << "~" << Number;
|
||||
if (ArgType->isSubClassOf("LLVMExtendedElementVectorType"))
|
||||
OS << "~(ExtendedElementVectorType | " << Number << ")";
|
||||
else if (ArgType->isSubClassOf("LLVMTruncatedElementVectorType"))
|
||||
OS << "~(TruncatedElementVectorType | " << Number << ")";
|
||||
else
|
||||
OS << "~" << Number;
|
||||
} else {
|
||||
MVT::SimpleValueType VT = getValueType(ArgType->getValueAsDef("VT"));
|
||||
OS << getEnumName(VT);
|
||||
@ -299,7 +311,12 @@ void IntrinsicEmitter::EmitVerifier(const std::vector<CodeGenIntrinsic> &Ints,
|
||||
if (ArgType->isSubClassOf("LLVMMatchType")) {
|
||||
unsigned Number = ArgType->getValueAsInt("Number");
|
||||
assert(Number < j + RetTys.size() && "Invalid matching number!");
|
||||
OS << "~" << Number;
|
||||
if (ArgType->isSubClassOf("LLVMExtendedElementVectorType"))
|
||||
OS << "~(ExtendedElementVectorType | " << Number << ")";
|
||||
else if (ArgType->isSubClassOf("LLVMTruncatedElementVectorType"))
|
||||
OS << "~(TruncatedElementVectorType | " << Number << ")";
|
||||
else
|
||||
OS << "~" << Number;
|
||||
} else {
|
||||
MVT::SimpleValueType VT = getValueType(ArgType->getValueAsDef("VT"));
|
||||
OS << getEnumName(VT);
|
||||
|
Loading…
Reference in New Issue
Block a user