6#include "mlir/IR/BuiltinTypes.h"
21 builder(cgm.
getBuilder()), theCXXABI(cgm.getCXXABI()),
22 theABIInfo(cgm.getTargetCIRGenInfo().
getABIInfo()) {}
25 for (
auto i = functionInfos.begin(), e = functionInfos.end(); i != e;)
30 return *builder.getContext();
50 return !
tagType->isIncompleteType();
62 if (
const auto *fpt = dyn_cast<FunctionProtoType>(ft))
63 for (
unsigned i = 0, e = fpt->getNumParams(); i != e; i++)
70mlir::Type CIRGenTypes::convertFunctionTypeInternal(
QualType qft) {
83 if (
const auto *fpt = dyn_cast<FunctionProtoType>(ft)) {
102 llvm::raw_svector_ostream outStream(typeName);
113 .
print(outStream, policy);
117 outStream << builder.getUniqueAnonRecordName();
122 return builder.getUniqueRecordName(std::string(typeName));
127 const auto it = recordDeclTypes.find(ty);
128 return it != recordDeclTypes.end() && it->second.isComplete();
135 llvm::SmallPtrSetImpl<const RecordDecl *> &alreadyChecked);
142 llvm::SmallPtrSetImpl<const RecordDecl *> &alreadyChecked) {
145 if (!alreadyChecked.insert(rd).second)
149 "Expect RecordDecl to be CompleteDefinition");
164 if (
const CXXRecordDecl *crd = dyn_cast<CXXRecordDecl>(rd)) {
167 ->castAs<RecordType>()
169 ->getDefinitionOrSelf(),
170 cgt, alreadyChecked))
188 llvm::SmallPtrSetImpl<const RecordDecl *> &alreadyChecked) {
191 qt = at->getValueType();
223 const Type *key = astContext.getCanonicalTagType(rd).getTypePtr();
224 cir::RecordType entry = recordDeclTypes[key];
231 entry = builder.getIncompleteRecordTy(name, rd);
232 recordDeclTypes[key] = entry;
241 deferredRecords.push_back(rd);
246 bool insertResult = recordsBeingLaidOut.insert(key).second;
248 assert(insertResult &&
"isSafeToCovert() should have caught this.");
251 if (
const auto *
cxxRecordDecl = dyn_cast<CXXRecordDecl>(rd)) {
253 if (base.isVirtual())
261 recordDeclTypes[key] = entry;
262 cirGenRecordLayouts[key] = std::move(layout);
265 bool eraseResult = recordsBeingLaidOut.erase(key);
267 assert(eraseResult &&
"record not in RecordsBeingLaidOut set?");
271 if (recordsBeingLaidOut.empty())
272 while (!deferredRecords.empty())
279 type = astContext.getCanonicalType(
type);
287 TypeCacheTy::iterator tci =
typeCache.find(ty);
294 mlir::Type resultType =
nullptr;
297 llvm_unreachable(
"Should have been handled above");
299 case Type::Builtin: {
302 case BuiltinType::Void:
303 resultType = cgm.voidTy;
307 case BuiltinType::Bool:
312 case BuiltinType::Char_S:
313 case BuiltinType::Int:
314 case BuiltinType::Int128:
315 case BuiltinType::Long:
316 case BuiltinType::LongLong:
317 case BuiltinType::SChar:
318 case BuiltinType::Short:
319 case BuiltinType::WChar_S:
326 case BuiltinType::SveInt8:
328 cir::VectorType::get(builder.getSInt8Ty(), 16,
true);
330 case BuiltinType::SveUint8:
332 cir::VectorType::get(builder.getUInt8Ty(), 16,
true);
334 case BuiltinType::SveInt16:
336 cir::VectorType::get(builder.getSInt16Ty(), 8,
true);
338 case BuiltinType::SveUint16:
340 cir::VectorType::get(builder.getUInt16Ty(), 8,
true);
342 case BuiltinType::SveFloat16:
343 resultType = cir::VectorType::get(builder.getFp16Ty(), 8,
346 case BuiltinType::SveBFloat16:
347 resultType = cir::VectorType::get(builder.getFp16Ty(), 8,
350 case BuiltinType::SveInt32:
352 cir::VectorType::get(builder.getSInt32Ty(), 4,
true);
354 case BuiltinType::SveUint32:
356 cir::VectorType::get(builder.getUInt32Ty(), 4,
true);
358 case BuiltinType::SveFloat32:
359 resultType = cir::VectorType::get(builder.getSingleTy(), 4,
362 case BuiltinType::SveInt64:
364 cir::VectorType::get(builder.getSInt64Ty(), 2,
true);
366 case BuiltinType::SveUint64:
368 cir::VectorType::get(builder.getUInt64Ty(), 2,
true);
370 case BuiltinType::SveFloat64:
371 resultType = cir::VectorType::get(builder.getDoubleTy(), 2,
374 case BuiltinType::SveBool:
375 resultType = cir::VectorType::get(builder.getUIntNTy(1), 16,
380 case BuiltinType::Char8:
381 case BuiltinType::Char16:
382 case BuiltinType::Char32:
383 case BuiltinType::Char_U:
384 case BuiltinType::UChar:
385 case BuiltinType::UInt:
386 case BuiltinType::UInt128:
387 case BuiltinType::ULong:
388 case BuiltinType::ULongLong:
389 case BuiltinType::UShort:
390 case BuiltinType::WChar_U:
397 case BuiltinType::Float16:
398 resultType = cgm.fP16Ty;
400 case BuiltinType::Half:
401 if (astContext.getLangOpts().NativeHalfType ||
402 !astContext.getTargetInfo().useFP16ConversionIntrinsics()) {
403 resultType = cgm.fP16Ty;
406 resultType = cgm.sInt32Ty;
409 case BuiltinType::BFloat16:
410 resultType = cgm.bFloat16Ty;
412 case BuiltinType::MFloat8:
413 resultType = cgm.uInt8Ty;
415 case BuiltinType::Float:
416 assert(&astContext.getFloatTypeSemantics(
type) ==
417 &llvm::APFloat::IEEEsingle() &&
418 "ClangIR NYI: 'float' in a format other than IEEE 32-bit");
419 resultType = cgm.floatTy;
421 case BuiltinType::Double:
422 assert(&astContext.getFloatTypeSemantics(
type) ==
423 &llvm::APFloat::IEEEdouble() &&
424 "ClangIR NYI: 'double' in a format other than IEEE 64-bit");
425 resultType = cgm.doubleTy;
427 case BuiltinType::LongDouble:
429 builder.getLongDoubleTy(astContext.getFloatTypeSemantics(
type));
431 case BuiltinType::Float128:
432 resultType = cgm.fP128Ty;
434 case BuiltinType::Ibm128:
436 resultType = cgm.sInt32Ty;
439 case BuiltinType::NullPtr:
444 resultType = builder.getVoidPtrTy();
449 resultType = cgm.sInt32Ty;
455 case Type::Complex: {
457 mlir::Type elementTy =
convertType(ct->getElementType());
458 resultType = cir::ComplexType::get(elementTy);
462 case Type::LValueReference:
463 case Type::RValueReference: {
467 resultType = builder.getPointerTo(pointeeType, elemTy.
getAddressSpace());
468 assert(resultType &&
"Cannot get pointer type?");
472 case Type::Pointer: {
479 resultType = builder.getPointerTo(pointeeType, elemTy.
getAddressSpace());
483 case Type::VariableArray: {
493 case Type::IncompleteArray: {
502 elemTy = cgm.sInt8Ty;
505 resultType = cir::ArrayType::get(elemTy, 0);
509 case Type::ConstantArray: {
517 resultType = cir::ArrayType::get(elemTy, arrTy->
getSize().getZExtValue());
521 case Type::ExtVector:
531 if (
auto integerType = ed->getIntegerType(); !integerType.isNull())
536 resultType = cgm.uInt32Ty;
540 case Type::MemberPointer: {
544 auto clsTy = mlir::cast<cir::RecordType>(
546 if (mpt->isMemberDataPointer()) {
547 mlir::Type memberTy =
convertType(mpt->getPointeeType());
548 resultType = cir::DataMemberType::get(memberTy, clsTy);
550 auto memberFuncTy =
getFunctionType(cgm.getTypes().arrangeCXXMethodType(
554 resultType = cir::MethodType::get(memberFuncTy, clsTy);
559 case Type::FunctionNoProto:
560 case Type::FunctionProto:
561 resultType = convertFunctionTypeInternal(
type);
566 if (bitIntTy->getNumBits() > cir::IntType::maxBitwidth()) {
568 resultType = cgm.sInt32Ty;
570 resultType = cir::IntType::get(&
getMLIRContext(), bitIntTy->getNumBits(),
571 bitIntTy->isSigned(),
582 uint64_t valueSize = astContext.getTypeSize(valueType);
583 uint64_t atomicSize = astContext.getTypeSize(ty);
584 if (valueSize != atomicSize) {
585 cgm.errorNYI(
"convertType: atomic type value size != atomic size");
593 type->getTypeClassName());
594 resultType = cgm.sInt32Ty;
598 assert(resultType &&
"Type conversion not yet implemented");
606 if (
qualType->isConstantMatrixType()) {
607 cgm.errorNYI(
"Matrix type conversion");
613 assert(!forBitField &&
"Bit fields NYI");
617 if (forBitField &&
qualType->isBitIntType())
618 assert(!
qualType->isBitIntType() &&
"Bit field with type _BitInt NYI");
620 return convertedType;
626 const auto *key = astContext.getCanonicalTagType(rd).getTypePtr();
629 auto it = cirGenRecordLayouts.find(key);
630 if (it != cirGenRecordLayouts.end())
637 it = cirGenRecordLayouts.find(key);
639 assert(it != cirGenRecordLayouts.end() &&
640 "Unable to find record layout information for type");
646 return astContext.getTargetNullPointerValue(t) == 0;
648 if (
const auto *at = astContext.getAsArrayType(t)) {
652 if (
const auto *cat = dyn_cast<ConstantArrayType>(at))
653 if (astContext.getConstantArrayElementCount(cat) == 0)
661 return theCXXABI.isZeroInitializable(mpt);
663 if (t->
getAs<HLSLInlineSpirvType>())
665 "isZeroInitializable for HLSLInlineSpirvType");
678 assert(llvm::all_of(argTypes,
681 llvm::FoldingSetNodeID id;
685 void *insertPos =
nullptr;
693 "Bad match based on CIRGenFunctionInfo folding set id");
702 functionInfos.InsertNode(fi, insertPos);
708 assert(!dyn_cast<ObjCMethodDecl>(gd.
getDecl()) &&
709 "This is reported as a FIXME in LLVM codegen");
725 if ([[maybe_unused]]
const auto *ed = dyn_cast<EnumDecl>(td)) {
733 ed->getASTContext().getCanonicalTagType(ed)->getTypePtr()) ||
735 typeCache[ed->getASTContext().getCanonicalTagType(ed)->getTypePtr()]));
745 if (rd->isDependentType())
750 if (recordDeclTypes.count(astContext.getCanonicalTagType(rd).getTypePtr()))
764 ? cgm.getDataLayout().getProgramAddressSpace()
Defines the clang::ASTContext interface.
static bool isSafeToConvert(QualType qt, CIRGenTypes &cgt, llvm::SmallPtrSetImpl< const RecordDecl * > &alreadyChecked)
Return true if it is safe to convert this field type, which requires the record elements contained by...
static Decl::Kind getKind(const Decl *D)
C Language Family Type Representation.
const ArrayType * getAsArrayType(QualType T) const
Type Query functions.
CanQualType getCanonicalTagType(const TagDecl *TD) const
unsigned getTargetAddressSpace(LangAS AS) const
QualType getElementType() const
unsigned getIndexTypeCVRQualifiers() const
CanQualType getReturnType() const
const_arg_iterator argTypesEnd() const
static CIRGenFunctionInfo * create(FunctionType::ExtInfo info, bool instanceMethod, CanQualType resultType, llvm::ArrayRef< CanQualType > argTypes, RequiredArgs required)
static void Profile(llvm::FoldingSetNodeID &id, bool instanceMethod, FunctionType::ExtInfo info, RequiredArgs required, CanQualType resultType, llvm::ArrayRef< CanQualType > argTypes)
const_arg_iterator argTypesBegin() const
This class organizes the cross-function state that is used while generating CIR code.
This class handles record and union layout info while lowering AST types to CIR types.
bool isZeroInitializable() const
Check whether this struct can be C++ zero-initialized with a zeroinitializer.
This class organizes the cross-module state that is used while lowering AST types to CIR types.
const CIRGenFunctionInfo & arrangeGlobalDeclaration(GlobalDecl gd)
unsigned getTargetAddressSpace(QualType ty) const
const CIRGenFunctionInfo & arrangeCXXStructorDeclaration(clang::GlobalDecl gd)
const CIRGenFunctionInfo & arrangeCIRFunctionInfo(CanQualType returnType, bool isInstanceMethod, llvm::ArrayRef< CanQualType > argTypes, FunctionType::ExtInfo info, RequiredArgs required)
const CIRGenFunctionInfo & arrangeFreeFunctionType(CanQual< FunctionProtoType > fpt)
bool isZeroInitializable(clang::QualType ty)
Return whether a type can be zero-initialized (in the C++ sense) with an LLVM zeroinitializer.
bool isFuncTypeConvertible(const clang::FunctionType *ft)
Utility to check whether a function type can be converted to a CIR type (i.e.
CIRGenTypes(CIRGenModule &cgm)
bool isRecordBeingLaidOut(const clang::Type *ty) const
CIRGenBuilderTy & getBuilder() const
mlir::MLIRContext & getMLIRContext() const
cir::FuncType getFunctionType(const CIRGenFunctionInfo &info)
Get the CIR function type for.
bool isFuncParamTypeConvertible(clang::QualType type)
Return true if the specified type in a function parameter or result position can be converted to a CI...
void updateCompletedType(const clang::TagDecl *td)
UpdateCompletedType - when we find the full definition for a TagDecl, replace the 'opaque' type we pr...
std::string getRecordTypeName(const clang::RecordDecl *, llvm::StringRef suffix)
bool noRecordsBeingLaidOut() const
const ABIInfo & getABIInfo() const
const CIRGenFunctionInfo & arrangeFunctionDeclaration(const clang::FunctionDecl *fd)
Free functions are functions that are compatible with an ordinary C function pointer type.
clang::ASTContext & getASTContext() const
bool isRecordLayoutComplete(const clang::Type *ty) const
Return true if the specified type is already completely laid out.
mlir::Type convertType(clang::QualType type)
Convert a Clang type into a mlir::Type.
const CIRGenRecordLayout & getCIRGenRecordLayout(const clang::RecordDecl *rd)
Return record layout info for the given record decl.
std::unique_ptr< CIRGenRecordLayout > computeRecordLayout(const clang::RecordDecl *rd, cir::RecordType *ty)
mlir::Type convertRecordDeclType(const clang::RecordDecl *recordDecl)
Lay out a tagged decl type like struct or union.
mlir::Type convertTypeForMem(clang::QualType, bool forBitField=false)
Convert type T into an mlir::Type.
A class for recording the number of arguments that a function signature requires.
Represents a base class of a C++ class.
Represents a C++ struct/union/class.
static CanQual< T > CreateUnsafe(QualType Other)
Builds a canonical type from a QualType.
bool isCanonicalAsParam() const
Determines if this canonical type is furthermore canonical as a parameter.
const T * getTypePtr() const
Retrieve the underlying type pointer, which refers to a canonical type.
Represents the canonical version of C arrays with a specified constant size.
llvm::APInt getSize() const
Return the constant array size as an APInt.
Represents a member of a struct/union/class.
Represents a K&R-style 'int foo()' function, which has no information available about its arguments.
Represents a prototype with parameter type info, e.g.
A class which abstracts out some details necessary for making a call.
FunctionType - C99 6.7.5.3 - Function Declarators.
QualType getReturnType() const
GlobalDecl - represents a global declaration.
const Decl * getDecl() const
Represents a C array with an unspecified size.
A pointer to member type per C++ 8.3.3 - Pointers to members.
Represents a C++ nested name specifier, such as "\::std::vector<int>::".
CXXRecordDecl * getAsRecordDecl() const
Retrieve the record declaration stored in this nested name specifier, or null.
const Type * getAsType() const
PointerType - C99 6.7.5.1 - Pointer Declarators.
QualType getPointeeType() const
A (possibly-)qualified type.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
LangAS getAddressSpace() const
Return the address space of this type.
void print(raw_ostream &OS, const PrintingPolicy &Policy, const Twine &PlaceHolder=Twine(), unsigned Indentation=0) const
bool hasAddressSpace() const
Check if this type has any address space qualifier.
Represents a struct/union/class.
field_range fields() const
RecordDecl * getDefinition() const
Returns the RecordDecl that actually defines this struct/union/class.
Base for LValueReferenceType and RValueReferenceType.
QualType getPointeeType() const
Encodes a location in the source.
Represents the declaration of a struct/union/class/enum.
bool isCompleteDefinition() const
Return true if this decl has its body fully specified.
The base class of the type hierarchy.
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isConstantMatrixType() const
EnumDecl * castAsEnumDecl() const
bool isFunctionType() const
TypeClass getTypeClass() const
const T * getAs() const
Member-template getAs<specific type>'.
Represents a C array with a specified size that is not an integer-constant-expression.
Represents a GCC generic vector type.
unsigned getNumElements() const
QualType getElementType() const
Defines the clang::TargetInfo interface.
bool isSized(mlir::Type ty)
Returns true if the type is a CIR sized type.
@ Type
The l-value was considered opaque, so the alignment was determined from a type.
const internal::VariadicDynCastAllOfMatcher< Decl, TypedefNameDecl > typedefNameDecl
Matches typedef name declarations.
const AstTypeMatcher< TagType > tagType
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const AstTypeMatcher< RecordType > recordType
const internal::VariadicDynCastAllOfMatcher< Decl, CXXRecordDecl > cxxRecordDecl
Matches C++ class declarations.
const internal::VariadicDynCastAllOfMatcher< Decl, RecordDecl > recordDecl
Matches class, struct, and union declarations.
const internal::VariadicAllOfMatcher< QualType > qualType
Matches QualTypes in the clang AST.
The JSON file list parser is used to communicate input to InstallAPI.
CanQual< Type > CanQualType
Represents a canonical, potentially-qualified type.
bool isa(CodeGen::Address addr)
bool isInstanceMethod(const Decl *D)
U cast(CodeGen::Address addr)
static bool opCallCallConv()
static bool generateDebugInfo()
Describes how types, statements, expressions, and declarations should be printed.
unsigned SuppressTagKeyword
Whether type printing should skip printing the tag keyword.
unsigned AlwaysIncludeTypeForTemplateArgument
Whether to use type suffixes (eg: 1U) on integral non-type template parameters.
unsigned SuppressInlineNamespace
Suppress printing parts of scope specifiers that correspond to inline namespaces.
unsigned PrintAsCanonical
Whether to print entities as written or canonically.