18#include "mlir/IR/Location.h"
24#include "llvm/ADT/ScopeExit.h"
25#include "llvm/IR/FPEnv.h"
32 bool suppressNewContext)
43 switch (
type->getTypeClass()) {
44#define TYPE(name, parent)
45#define ABSTRACT_TYPE(name, parent)
46#define NON_CANONICAL_TYPE(name, parent) case Type::name:
47#define DEPENDENT_TYPE(name, parent) case Type::name:
48#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(name, parent) case Type::name:
49#include "clang/AST/TypeNodes.inc"
50 llvm_unreachable(
"non-canonical or dependent type in IR-generation");
53 case Type::DeducedTemplateSpecialization:
54 llvm_unreachable(
"undeduced type in IR-generation");
59 case Type::BlockPointer:
60 case Type::LValueReference:
61 case Type::RValueReference:
62 case Type::MemberPointer:
65 case Type::ConstantMatrix:
66 case Type::FunctionProto:
67 case Type::FunctionNoProto:
69 case Type::ObjCObjectPointer:
72 case Type::OverflowBehavior:
73 case Type::HLSLAttributedResource:
74 case Type::HLSLInlineSpirv:
82 case Type::ConstantArray:
83 case Type::IncompleteArray:
84 case Type::VariableArray:
86 case Type::ObjCObject:
87 case Type::ObjCInterface:
88 case Type::ArrayParameter:
96 llvm_unreachable(
"unknown type kind!");
101 return cgm.getTypes().convertTypeForMem(t);
105 return cgm.getTypes().convertType(t);
115 return mlir::FileLineColLoc::get(builder.getStringAttr(filename),
121 assert(
currSrcLoc &&
"expected to inherit some source location");
125 return builder.getUnknownLoc();
135 mlir::Attribute metadata;
141 assert(
currSrcLoc &&
"expected to inherit some source location");
145 return builder.getUnknownLoc();
150 mlir::Attribute metadata;
177 ignoreCaseStmts =
true;
180 return std::any_of(
s->child_begin(),
s->child_end(),
181 [=](
const Stmt *subStmt) {
182 return containsLabel(subStmt, ignoreCaseStmts);
191 llvm::APSInt resultInt;
195 resultBool = resultInt.getBoolValue();
203 llvm::APSInt &resultInt,
211 llvm::APSInt intValue = result.
Val.
getInt();
215 resultInt = intValue;
219void CIRGenFunction::emitAndUpdateRetAlloca(
QualType type, mlir::Location loc,
221 if (!
type->isVoidType()) {
228void CIRGenFunction::declare(mlir::Value addrVal,
const Decl *var, QualType ty,
229 mlir::Location loc, CharUnits alignment,
232 assert(!
symbolTable.count(var) &&
"not supposed to be available just yet");
234 Address addr(addrVal, alignment);
235 cir::AllocaOp allocaOp = addr.getUnderlyingAllocaOp();
236 assert(allocaOp &&
"expected cir::AllocaOp");
240 if (ty->isReferenceType() || ty.isConstQualified())
256 for (mlir::Block *retBlock : localScope->
getRetBlocks()) {
257 mlir::OpBuilder::InsertionGuard guard(builder);
258 builder.setInsertionPointToEnd(retBlock);
259 retBlocks.push_back(retBlock);
260 mlir::Location retLoc = localScope->
getRetLoc(retBlock);
272 mlir::Block *curBlock = builder.getBlock();
275 if (curBlock->mightHaveTerminator() && curBlock->getTerminator())
285 if (mlir::Region *scopeRegion = entryBlock->getParent();
286 scopeRegion && !scopeRegion->empty() &&
287 curBlock->getParent() != scopeRegion) {
288 builder.setInsertionPointToEnd(&scopeRegion->back());
289 curBlock = builder.getBlock();
290 if (curBlock->mightHaveTerminator() && curBlock->getTerminator())
297 bool isEntryBlock = builder.getInsertionBlock()->isEntryBlock();
298 if (!isEntryBlock && curBlock->empty()) {
300 for (mlir::Block *retBlock : retBlocks) {
301 if (retBlock->getUses().empty())
307 if (localScope->
depth == 0) {
311 mlir::Location retLoc = localScope->
getRetLoc(retBlock);
312 if (retBlock->getUses().empty()) {
315 cir::BrOp::create(builder, retLoc, retBlock);
319 emitImplicitReturn();
326 if (!localScope->
isTernary() && !curBlock->mightHaveTerminator()) {
327 !retVal ? cir::YieldOp::create(builder, localScope->endLoc)
328 : cir::YieldOp::create(builder, localScope->endLoc, retVal);
332cir::ReturnOp CIRGenFunction::LexicalScope::emitReturn(mlir::Location loc) {
335 auto fn = dyn_cast<cir::FuncOp>(cgf.curFn);
336 assert(fn &&
"emitReturn from non-function");
338 if (!fn.getFunctionType().hasVoidReturn()) {
340 auto value = cir::LoadOp::create(
341 builder, loc, fn.getFunctionType().getReturnType(), *cgf.fnRetAlloca);
342 return cir::ReturnOp::create(builder, loc,
345 return cir::ReturnOp::create(builder, loc);
355 return classDecl->hasTrivialDestructor();
362 mlir::Operation *op = &block->back();
363 auto cleanupScopeOp = mlir::dyn_cast<cir::CleanupScopeOp>(op);
371 for (mlir::Block &bodyBlock : cleanupScopeOp.getBodyRegion()) {
372 if (bodyBlock.mightHaveTerminator()) {
373 if (mlir::isa<cir::YieldOp>(bodyBlock.getTerminator()))
375 assert(!mlir::isa<cir::BreakOp>(bodyBlock.getTerminator()) &&
376 !mlir::isa<cir::ContinueOp>(bodyBlock.getTerminator()) &&
377 !mlir::isa<cir::ResumeOp>(bodyBlock.getTerminator()));
383void CIRGenFunction::LexicalScope::emitImplicitReturn() {
384 CIRGenBuilderTy &builder = cgf.getBuilder();
385 LexicalScope *localScope = cgf.curLexScope;
394 if (cgf.getLangOpts().CPlusPlus && !fd->hasImplicitReturnZero() &&
395 !cgf.sawAsmBlock && !fd->getReturnType()->isVoidType() &&
396 builder.getInsertionBlock() &&
398 bool shouldEmitUnreachable =
399 cgf.cgm.getCodeGenOpts().StrictReturn ||
402 if (shouldEmitUnreachable) {
404 if (cgf.cgm.getCodeGenOpts().OptimizationLevel == 0)
405 cir::TrapOp::create(builder, localScope->endLoc);
407 cir::UnreachableOp::create(builder, localScope->endLoc);
408 builder.clearInsertionPoint();
413 (void)emitReturn(localScope->endLoc);
421 scope = scope->parentScope;
434 if (value.getType() == ty)
437 assert((mlir::isa<cir::IntType>(ty) || cir::isAnyFloatingPointType(ty)) &&
438 "unexpected promotion type");
440 if (mlir::isa<cir::IntType>(ty))
441 return cgf.
getBuilder().CIRBaseBuilderTy::createIntCast(value, ty);
447 mlir::Block *entryBB,
451 if (fd && fd->
hasAttr<NakedAttr>()) {
452 cgm.errorNYI(bodyBeginLoc,
"naked function decl");
456 for (
const auto nameValue : llvm::zip(args, entryBB->getArguments())) {
457 const VarDecl *paramVar = std::get<0>(nameValue);
458 mlir::Value paramVal = std::get<1>(nameValue);
461 paramVal.setLoc(paramLoc);
463 mlir::Value addrVal =
468 declare(addrVal, paramVar, paramVar->
getType(), paramLoc, alignment,
481 mlir::Location fnBodyBegin =
getLoc(bodyBeginLoc);
482 builder.CIRBaseBuilderTy::createStore(fnBodyBegin, paramVal, addrVal);
484 assert(builder.getInsertionBlock() &&
"Should be valid");
488 cir::FuncOp fn, cir::FuncType funcType,
492 "CIRGenFunction can only be used for one function at a time");
500 const auto *fd = dyn_cast_or_null<FunctionDecl>(d);
505 mlir::Block *entryBB = &fn.getBlocks().front();
506 builder.setInsertionPointToStart(entryBB);
512 if (
Stmt *body = fd->getBody())
513 bodyBeginLoc = body->getBeginLoc();
515 bodyBeginLoc = fd->getLocation();
527 if (
Stmt *body = fd->getBody())
528 bodyEndLoc = body->getEndLoc();
530 bodyEndLoc = fd->getLocation();
532 emitAndUpdateRetAlloca(returnType,
getLoc(bodyEndLoc),
533 getContext().getTypeAlignInChars(returnType));
542 if (fd && fd->hasImplicitReturnZero()) {
544 mlir::Location bodyBeginMLIRLoc =
getLoc(bodyBeginLoc);
545 mlir::Value zero = builder.getNullValue(cirRetTy, bodyBeginMLIRLoc);
546 builder.CIRBaseBuilderTy::createStore(bodyBeginMLIRLoc, zero,
555 if (isa_and_nonnull<CXXMethodDecl>(d) &&
557 cgm.getCXXABI().emitInstanceFunctionProlog(loc, *
this);
560 if (md->getParent()->isLambda() && md->getOverloadedOperator() == OO_Call) {
562 auto fn = dyn_cast<cir::FuncOp>(
curFn);
563 assert(fn &&
"lambda in non-function region");
589 for (
auto *fd : md->getParent()->fields()) {
590 if (fd->hasCapturedVLAType())
591 cgm.errorNYI(loc,
"lambda captured VLA type");
607 if (!fd || !fd->hasAttr<NakedAttr>()) {
608 for (
const VarDecl *vd : args) {
613 if (
const auto *pvd = dyn_cast<ParmVarDecl>(vd))
614 ty = pvd->getOriginalType();
624 for (cir::BlockAddressOp &blockAddress :
cgm.unresolvedBlockAddressToLabel) {
625 cir::LabelOp labelOp =
626 cgm.lookupBlockAddressInfo(blockAddress.getBlockAddrInfo());
627 assert(labelOp &&
"expected cir.labelOp to already be emitted");
628 cgm.updateResolvedBlockAddress(blockAddress, labelOp);
630 cgm.unresolvedBlockAddressToLabel.clear();
638 mlir::OpBuilder::InsertionGuard guard(builder);
640 for (
auto &[blockAdd, labelOp] :
cgm.blockAddressToLabel) {
641 succesors.push_back(labelOp->getBlock());
642 rangeOperands.push_back(labelOp->getBlock()->getArguments());
644 cir::IndirectBrOp::create(builder, builder.getUnknownLoc(),
646 rangeOperands, succesors);
647 cgm.blockAddressToLabel.clear();
662 indrBr.setPoison(
true);
678 "deferred conditional cleanups were not consumed by a "
679 "FullExprCleanupScope");
686 if (
const CompoundStmt *block = dyn_cast<CompoundStmt>(body))
697 for (mlir::Block &block : func.getBlocks()) {
698 if (block.empty() && block.getUses().empty())
699 blocksToDelete.push_back(&block);
701 for (mlir::Block *block : blocksToDelete)
706 cir::FuncType funcType) {
710 if (funcDecl->isInlineBuiltinDeclaration()) {
714 std::string fdInlineName = (
cgm.getMangledName(funcDecl) +
".inline").str();
716 mlir::cast_or_null<cir::FuncOp>(
cgm.getGlobalValue(fdInlineName));
718 mlir::OpBuilder::InsertionGuard guard(builder);
719 builder.setInsertionPoint(fn);
720 clone = cir::FuncOp::create(builder, fn.getLoc(), fdInlineName,
721 fn.getFunctionType());
722 cgm.insertGlobalSymbol(clone);
723 clone.setLinkage(cir::GlobalLinkageKind::InternalLinkage);
724 clone.setSymVisibility(
"private");
725 clone.setInlineKind(cir::InlineKind::AlwaysInline);
727 fn.setLinkage(cir::GlobalLinkageKind::ExternalLinkage);
728 fn.setSymVisibility(
"private");
736 if (LLVM_UNLIKELY(pd->isInlineBuiltinDeclaration())) {
737 std::string inlineName = funcDecl->getName().str() +
".inline";
738 if (
auto inlineFn = mlir::cast_or_null<cir::FuncOp>(
739 cgm.getGlobalValue(inlineName))) {
744 .replaceAllSymbolUses(fn.getSymNameAttr(),
cgm.getModule())
746 llvm_unreachable(
"Failed to replace inline builtin symbol uses");
747 cgm.eraseGlobalSymbol(inlineFn);
756 Stmt *body = funcDecl->getBody();
761 : builder.getUnknownLoc()};
764 return clangLoc.isValid() ?
getLoc(clangLoc) : builder.getUnknownLoc();
766 const mlir::Location fusedLoc = mlir::FusedLoc::get(
768 {validMLIRLoc(bodyRange.
getBegin()), validMLIRLoc(bodyRange.
getEnd())});
769 mlir::Block *entryBB = fn.addEntryBlock();
783 if (body && isa_and_nonnull<CoroutineBodyStmt>(body))
784 llvm::append_range(
fnArgs, funcDecl->parameters());
791 funcDecl->hasAttr<CUDAGlobalAttr>()) {
792 cgm.getCUDARuntime().emitDeviceStub(*
this, fn, args);
812 llvm_unreachable(
"no definition for normal function");
815 if (mlir::failed(fn.verifyBody()))
830 assert((
cgm.getTarget().getCXXABI().hasConstructorVariants() ||
832 "can only generate complete ctor for this ABI");
837 cgm.getTarget().getCXXABI().hasConstructorVariants()) {
843 Stmt *body = ctor->getBody(definition);
844 assert(definition == ctor &&
"emitting wrong constructor body");
846 bool isTryBody = isa_and_nonnull<CXXTryStmt>(body);
855 Stmt *emitterBody =
nullptr;
858 : ctor(ctor), ctorType(ctorType), args(args),
860 ~ctorTryBodyEmitter()
override =
default;
870 return cgf.
emitStmt(emitterBody,
true);
874 ctorTryBodyEmitter emitter{ctor, ctorType, args, isTryBody, body};
875 mlir::LogicalResult bodyRes =
881 if (bodyRes.failed())
882 cgm.errorNYI(ctor->getSourceRange(),
883 "emitConstructorBody: emit body statement failed.");
927 const bool isTryBody = isa_and_nonnull<CXXTryStmt>(body);
943 llvm_unreachable(
"not expecting a unified dtor");
945 llvm_unreachable(
"not expecting a COMDAT");
948 llvm_unreachable(
"already handled deleting case");
951 assert((body ||
getTarget().getCXXABI().isMicrosoft()) &&
952 "can't emit a dtor without a body for non-Microsoft ABIs");
980 assert(dtor->
isImplicit() &&
"bodyless dtor not implicit");
1006 CharUnits align =
cgm.getNaturalTypeAlignment(ty, &baseInfo);
1013 CharUnits alignment =
cgm.getNaturalTypeAlignment(ty, &baseInfo);
1021static llvm::fp::ExceptionBehavior
1025 return llvm::fp::ebIgnore;
1027 return llvm::fp::ebMayTrap;
1029 return llvm::fp::ebStrict;
1031 llvm_unreachable(
"expected explicitly initialized exception behavior");
1033 llvm_unreachable(
"unsupported FP exception behavior");
1039 QualType retTy = fd->getReturnType();
1044 const auto *md = dyn_cast<CXXMethodDecl>(fd);
1045 if (md && md->isImplicitObjectMemberFunction()) {
1046 if (
cgm.getCXXABI().hasThisReturn(gd))
1047 cgm.errorNYI(fd->getSourceRange(),
"this return");
1048 else if (
cgm.getCXXABI().hasMostDerivedReturn(gd))
1049 cgm.errorNYI(fd->getSourceRange(),
"most derived return");
1050 cgm.getCXXABI().buildThisParam(*
this, args);
1053 bool passedParams =
true;
1054 if (
const auto *cd = dyn_cast<CXXConstructorDecl>(fd))
1055 if (
auto inherited = cd->getInheritedConstructor())
1060 for (
auto *param : fd->parameters()) {
1061 args.push_back(param);
1062 if (!param->hasAttr<PassObjectSizeAttr>())
1066 getContext(), param->getDeclContext(), param->getLocation(),
1069 args.push_back(implicit);
1074 cgm.getCXXABI().addImplicitStructorParams(*
this, retTy, args);
1085 assert(e->
isTransparent() &&
"non-transparent glvalue init list");
1089static std::variant<LValue, RValue>
1094 llvm::scope_exit opaque_cleanup{
1095 [&]() { llvm::for_each(opaques, [&](OVMD &o) { o.unbind(cgf); }); }};
1099 std::variant<LValue, RValue> result;
1104 if (
const auto *ov = dyn_cast<OpaqueValueExpr>(semantic)) {
1107 if (ov->isUnique()) {
1108 assert(ov != resultExpr &&
1109 "A unique OVE cannot be used as the result expression");
1116 if (ov == resultExpr && ov->
isPRValue() && !forLValue &&
1119 "emitPseudoObjectExpr for RValue & aggregate kind");
1121 opaqueData = OVMD::bind(cgf, ov, ov->getSourceExpr());
1124 if (ov == resultExpr) {
1128 "emitPseudoObjectExpr as result");
1133 "emitPseudoObjectExpr as an RValue");
1136 opaques.push_back(opaqueData);
1137 }
else if (semantic == resultExpr) {
1148 "emitPseudoObjectExpr as an ignored value");
1159 return std::get<RValue>(
1176 std::string(
"l-value not implemented for '") +
1179 case Expr::ConditionalOperatorClass:
1181 case Expr::BinaryConditionalOperatorClass:
1183 case Expr::ArraySubscriptExprClass:
1185 case Expr::ExtVectorElementExprClass:
1187 case Expr::UnaryOperatorClass:
1189 case Expr::StringLiteralClass:
1191 case Expr::MemberExprClass:
1193 case Expr::CompoundLiteralExprClass:
1195 case Expr::PredefinedExprClass:
1197 case Expr::BinaryOperatorClass:
1199 case Expr::CompoundAssignOperatorClass: {
1203 "CompoundAssignOperator with AtomicType");
1211 case Expr::CallExprClass:
1212 case Expr::CXXMemberCallExprClass:
1213 case Expr::CXXOperatorCallExprClass:
1214 case Expr::UserDefinedLiteralClass:
1216 case Expr::ExprWithCleanupsClass: {
1233 case Expr::CXXDefaultArgExprClass: {
1238 case Expr::CXXTypeidExprClass:
1240 case Expr::ParenExprClass:
1242 case Expr::GenericSelectionExprClass:
1244 case Expr::DeclRefExprClass:
1246 case Expr::ImplicitCastExprClass:
1247 case Expr::CStyleCastExprClass:
1248 case Expr::CXXStaticCastExprClass:
1249 case Expr::CXXDynamicCastExprClass:
1250 case Expr::CXXReinterpretCastExprClass:
1251 case Expr::CXXConstCastExprClass:
1252 case Expr::CXXFunctionalCastExprClass:
1256 case Expr::MaterializeTemporaryExprClass:
1258 case Expr::OpaqueValueExprClass:
1260 case Expr::ChooseExprClass:
1262 case Expr::SubstNonTypeTemplateParmExprClass:
1264 case Expr::InitListExprClass:
1266 case Expr::PseudoObjectExprClass:
1268 case Expr::CXXDefaultInitExprClass: {
1278 llvm::raw_svector_ostream
out(buffer);
1280 return std::string(
out.str());
1300 cgm.errorNYI(loc,
"Cast the dest ptr to the appropriate i8 pointer type");
1309 "emitNullInitialization for zero size VariableArrayType");
1319 if (!
cgm.getTypes().isZeroInitializable(ty)) {
1320 cgm.errorNYI(loc,
"type is not zero initializable");
1327 const mlir::Value zeroValue = builder.getNullValue(
convertType(ty), loc);
1328 builder.createStore(loc, zeroValue, destPtr);
1340 ConstructorHelper(fpFeatures);
1343void CIRGenFunction::CIRGenFPOptionsRAII::ConstructorHelper(
1345 oldFPFeatures = cgf.curFPFeatures;
1346 cgf.curFPFeatures = fpFeatures;
1348 oldExcept = cgf.builder.getDefaultConstrainedExcept();
1349 oldRounding = cgf.builder.getDefaultConstrainedRounding();
1351 if (oldFPFeatures == fpFeatures)
1357 [[maybe_unused]] llvm::RoundingMode newRoundingBehavior =
1360 [[maybe_unused]] llvm::fp::ExceptionBehavior newExceptionBehavior =
1368 assert((cgf.curFuncDecl ==
nullptr || cgf.builder.getIsFPConstrained() ||
1371 (newExceptionBehavior == llvm::fp::ebIgnore &&
1372 newRoundingBehavior == llvm::RoundingMode::NearestTiesToEven)) &&
1373 "FPConstrained should be enabled on entire function");
1380 cgf.curFPFeatures = oldFPFeatures;
1381 cgf.builder.setDefaultConstrainedExcept(oldExcept);
1382 cgf.builder.setDefaultConstrainedRounding(oldRounding);
1389 if (ce->
getCastKind() == CK_UncheckedDerivedToBase)
1399 if (ice->isGLValue())
1415 mlir::Value numVLAElements =
nullptr;
1427 baseType = elementType;
1428 return numVLAElements;
1441 uint64_t countFromCLAs = 1;
1444 auto cirArrayType = mlir::dyn_cast<cir::ArrayType>(addr.
getElementType());
1446 while (cirArrayType) {
1448 countFromCLAs *= cirArrayType.getSize();
1452 mlir::dyn_cast<cir::ArrayType>(cirArrayType.getElementType());
1456 "CIR and Clang types are out-of-sync");
1463 cgm.errorNYI(*
currSrcLoc,
"length for non-array underlying types");
1468 mlir::Value numElements =
1474 builder.createMul(numVLAElements.getLoc(), numVLAElements, numElements,
1485 mlir::OpBuilder::InsertionGuard guard(builder);
1487 builder.createBlock(builder.getBlock()->getParent(), {}, {voidPtrTy},
1488 {builder.getUnknownLoc()});
1493 SourceLocation assumptionLoc, int64_t alignment, mlir::Value offsetValue) {
1495 mlir::Location assumeLoc =
getLoc(assumptionLoc);
1496 mlir::Value alignValue = builder.getUInt64(alignment, assumeLoc);
1497 mlir::Value cond = builder.getBool(
true, assumeLoc);
1500 bundleArgs.push_back(offsetValue);
1501 cir::AssumeOp::create(builder, assumeLoc, cond, cir::AssumeBundleKind::Align,
1508 int64_t alignment, mlir::Value offsetValue) {
1517 cgm.getASTContext().getAsVariableArrayType(
type);
1518 assert(vla &&
"type was not a variable array type!");
1525 mlir::Value numElements;
1529 elementType =
type->getElementType();
1531 assert(vlaSize &&
"no size for VLA!");
1532 assert(vlaSize.getType() ==
sizeTy);
1535 numElements = vlaSize;
1541 builder.createMul(numElements.getLoc(), numElements, vlaSize,
1544 }
while ((
type =
getContext().getAsVariableArrayType(elementType)));
1546 assert(numElements &&
"Undefined elements number");
1547 return {numElements, elementType};
1553 assert(vlaSize &&
"no size for VLA!");
1554 assert(vlaSize.getType() ==
sizeTy);
1561 assert(
type->isVariablyModifiedType() &&
1562 "Must pass variably modified type to EmitVLASizes!");
1567 assert(
type->isVariablyModifiedType());
1569 const Type *ty =
type.getTypePtr();
1571 case Type::CountAttributed:
1572 case Type::PackIndexing:
1573 case Type::ArrayParameter:
1574 case Type::HLSLAttributedResource:
1575 case Type::HLSLInlineSpirv:
1576 case Type::PredefinedSugar:
1577 cgm.errorNYI(
"CIRGenFunction::emitVariablyModifiedType");
1580#define TYPE(Class, Base)
1581#define ABSTRACT_TYPE(Class, Base)
1582#define NON_CANONICAL_TYPE(Class, Base)
1583#define DEPENDENT_TYPE(Class, Base) case Type::Class:
1584#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base)
1585#include "clang/AST/TypeNodes.inc"
1587 "dependent type must be resolved before the CIR codegen");
1593 case Type::ExtVector:
1594 case Type::ConstantMatrix:
1598 case Type::TemplateSpecialization:
1599 case Type::ObjCTypeParam:
1600 case Type::ObjCObject:
1601 case Type::ObjCInterface:
1602 case Type::ObjCObjectPointer:
1604 case Type::OverflowBehavior:
1605 llvm_unreachable(
"type class is never variably-modified!");
1607 case Type::Adjusted:
1619 case Type::BlockPointer:
1623 case Type::LValueReference:
1624 case Type::RValueReference:
1628 case Type::MemberPointer:
1632 case Type::ConstantArray:
1633 case Type::IncompleteArray:
1638 case Type::VariableArray: {
1655 entry = builder.createIntCast(size,
sizeTy);
1662 case Type::FunctionProto:
1663 case Type::FunctionNoProto:
1669 case Type::UnaryTransform:
1670 case Type::Attributed:
1671 case Type::BTFTagAttributed:
1672 case Type::SubstTemplateTypeParm:
1673 case Type::MacroQualified:
1679 case Type::Decltype:
1681 case Type::DeducedTemplateSpecialization:
1685 case Type::TypeOfExpr:
1698 }
while (
type->isVariablyModifiedType());
1702 if (
getContext().getBuiltinVaListType()->isArrayType())
Defines the clang::Expr interface and subclasses for C++ expressions.
*collection of selector each with an associated kind and an ordered *collection of selectors A selector has a kind
__device__ __2f16 float __ockl_bool s
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
SourceManager & getSourceManager()
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
const ArrayType * getAsArrayType(QualType T) const
Type Query functions.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Address withPointer(mlir::Value newPtr) const
Return address with different pointer, but same element type and alignment.
mlir::Value getPointer() const
mlir::Type getElementType() const
static AggValueSlot ignored()
Returns an aggregate value slot indicating that the aggregate value is being ignored.
mlir::Value createFloatingCast(mlir::Value v, mlir::Type destType)
CIRGenFPOptionsRAII(CIRGenFunction &cgf, FPOptions FPFeatures)
The scope of a CXXDefaultInitExpr.
void exit(ArrayRef< mlir::Value * > valuesToReload={})
A non-RAII class containing all the information about a bound opaque value.
Enters a new scope for capturing cleanups, all of which will be executed once the scope is exited.
void forceCleanup(ArrayRef< mlir::Value * > valuesToReload={})
Force the emission of cleanups now, instead of waiting until this object is destroyed.
static bool isConstructorDelegationValid(const clang::CXXConstructorDecl *ctor)
Checks whether the given constructor is a valid subject for the complete-to-base constructor delegati...
void emitFunctionProlog(const FunctionArgList &args, mlir::Block *entryBB, const FunctionDecl *fd, SourceLocation bodyBeginLoc)
Emit the function prologue: declare function arguments in the symbol table.
mlir::Type convertType(clang::QualType t)
LValue emitOpaqueValueLValue(const OpaqueValueExpr *e)
static cir::TypeEvaluationKind getEvaluationKind(clang::QualType type)
Return the cir::TypeEvaluationKind of QualType type.
clang::GlobalDecl curGD
The GlobalDecl for the current function being compiled or the global variable currently being initial...
EHScopeStack::stable_iterator prologueCleanupDepth
The cleanup depth enclosing all the cleanups associated with the parameters.
cir::FuncOp generateCode(clang::GlobalDecl gd, cir::FuncOp fn, cir::FuncType funcType)
CIRGenTypes & getTypes() const
Address emitPointerWithAlignment(const clang::Expr *expr, LValueBaseInfo *baseInfo=nullptr)
Given an expression with a pointer type, emit the value and compute our best estimate of the alignmen...
void emitVariablyModifiedType(QualType ty)
RValue emitLoadOfLValue(LValue lv, SourceLocation loc)
Given an expression that represents a value lvalue, this method emits the address of the lvalue,...
const clang::LangOptions & getLangOpts() const
mlir::Value cxxStructorImplicitParamValue
void emitTrap(mlir::Location loc, bool createNewBlock)
Emit a trap instruction, which is used to abort the program in an abnormal way, usually for debugging...
VlaSizePair getVLASize(const VariableArrayType *type)
Returns an MLIR::Value+QualType pair that corresponds to the size, in non-variably-sized elements,...
LValue makeNaturalAlignPointeeAddrLValue(mlir::Value v, clang::QualType t)
Given a value of type T* that may not be to a complete object, construct an l-vlaue withi the natural...
LValue emitMemberExpr(const MemberExpr *e)
const TargetInfo & getTarget() const
LValue emitConditionalOperatorLValue(const AbstractConditionalOperator *expr)
LValue emitLValue(const clang::Expr *e)
Emit code to compute a designator that specifies the location of the expression.
const clang::Decl * curFuncDecl
Address loadCXXThisAddress()
LValue emitLValueForLambdaField(const FieldDecl *field)
std::string getCounterRefTmpAsString()
LValue makeNaturalAlignAddrLValue(mlir::Value val, QualType ty)
llvm::DenseMap< const Expr *, mlir::Value > vlaSizeMap
bool constantFoldsToSimpleInteger(const clang::Expr *cond, llvm::APSInt &resultInt, bool allowLabels=false)
If the specified expression does not fold to a constant, or if it does fold but contains a label,...
LValue emitComplexCompoundAssignmentLValue(const CompoundAssignOperator *e)
mlir::Location getLoc(clang::SourceLocation srcLoc)
Helpers to convert Clang's SourceLocation to a MLIR Location.
bool constantFoldsToBool(const clang::Expr *cond, bool &resultBool, bool allowLabels=false)
If the specified expression does not fold to a constant, or if it does but contains a label,...
void emitDelegateCXXConstructorCall(const clang::CXXConstructorDecl *ctor, clang::CXXCtorType ctorType, const FunctionArgList &args, clang::SourceLocation loc)
VlaSizePair getVLAElements1D(const VariableArrayType *vla)
Return the number of elements for a single dimension for the given array type.
mlir::Value emitArrayLength(const clang::ArrayType *arrayType, QualType &baseType, Address &addr)
Computes the length of an array in elements, as well as the base element type and a properly-typed fi...
void emitNullInitialization(mlir::Location loc, Address destPtr, QualType ty)
LValue emitArraySubscriptExpr(const clang::ArraySubscriptExpr *e)
llvm::ScopedHashTableScope< const clang::Decl *, mlir::Value > SymTableScopeTy
mlir::Block * indirectGotoBlock
IndirectBranch - The first time an indirect goto is seen we create a block reserved for the indirect ...
mlir::Operation * curFn
The current function or global initializer that is generated code for.
EHScopeStack ehStack
Tracks function scope overall cleanup handling.
void enterDtorCleanups(const CXXDestructorDecl *dtor, CXXDtorType type)
Enter the cleanups necessary to complete the given phase of destruction for a destructor.
llvm::SmallVector< const ParmVarDecl * > fnArgs
Save Parameter Decl for coroutine.
llvm::SmallVector< PendingCleanupEntry > deferredConditionalCleanupStack
std::optional< mlir::Value > fnRetAlloca
The compiler-generated variable that holds the return value.
void emitImplicitAssignmentOperatorBody(FunctionArgList &args)
mlir::Type convertTypeForMem(QualType t)
mlir::LogicalResult emitCXXTryStmt(const clang::CXXTryStmt &s, cxxTryBodyEmitter &bodyCallback)
clang::QualType buildFunctionArgList(clang::GlobalDecl gd, FunctionArgList &args)
void emitCtorPrologue(const clang::CXXConstructorDecl *ctor, clang::CXXCtorType ctorType, FunctionArgList &args)
This routine generates necessary code to initialize base classes and non-static data members belongin...
mlir::Value emitAlloca(llvm::StringRef name, mlir::Type ty, mlir::Location loc, clang::CharUnits alignment, bool insertIntoFnEntryBlock, mlir::Value arraySize=nullptr)
LValue emitAggExprToLValue(const Expr *e)
LValue emitCompoundAssignmentLValue(const clang::CompoundAssignOperator *e)
Address returnValue
The temporary alloca to hold the return value.
static bool hasAggregateEvaluationKind(clang::QualType type)
void finishFunction(SourceLocation endLoc)
mlir::LogicalResult emitFunctionBody(const clang::Stmt *body)
std::string getCounterAggTmpAsString()
LValue emitUnaryOpLValue(const clang::UnaryOperator *e)
clang::FieldDecl * lambdaThisCaptureField
const clang::Decl * curCodeDecl
This is the inner-most code context, which includes blocks.
void emitConstructorBody(FunctionArgList &args)
LValue emitCallExprLValue(const clang::CallExpr *e)
bool haveInsertPoint() const
True if an insertion point is defined.
void finishIndirectBranch()
LValue emitStringLiteralLValue(const StringLiteral *e, llvm::StringRef name=".str")
llvm::SmallDenseMap< const ParmVarDecl *, const ImplicitParamDecl * > sizeArguments
If a ParmVarDecl had the pass_object_size attribute, this will contain a mapping from said ParmVarDec...
void resolveBlockAddresses()
mlir::Value emitScalarExpr(const clang::Expr *e, bool ignoreResultAssign=false)
Emit the computation of the specified expression of scalar type.
LValue emitPseudoObjectLValue(const PseudoObjectExpr *E)
void popCleanupBlocks(EHScopeStack::stable_iterator oldCleanupStackDepth, ArrayRef< mlir::Value * > valuesToReload={})
Takes the old cleanup stack size and emits the cleanup blocks that have been added.
bool shouldNullCheckClassCastValue(const CastExpr *ce)
CIRGenBuilderTy & getBuilder()
bool didCallStackSave
Whether a cir.stacksave operation has been added.
LValue emitBinaryOperatorLValue(const BinaryOperator *e)
void startFunction(clang::GlobalDecl gd, clang::QualType returnType, cir::FuncOp fn, cir::FuncType funcType, FunctionArgList args, clang::SourceLocation loc, clang::SourceLocation startLoc)
Emit code for the start of a function.
CIRGenModule & getCIRGenModule()
unsigned counterRefTmp
Hold counters for incrementally naming temporaries.
mlir::MLIRContext & getMLIRContext()
void emitDestructorBody(FunctionArgList &args)
Emits the body of the current destructor.
LValue emitInitListLValue(const InitListExpr *e)
LValue emitCastLValue(const CastExpr *e)
Casts are never lvalues unless that cast is to a reference type.
LValue emitCXXTypeidLValue(const CXXTypeidExpr *e)
bool containsLabel(const clang::Stmt *s, bool ignoreCaseStmts=false)
Return true if the statement contains a label in it.
RValue emitPseudoObjectRValue(const PseudoObjectExpr *e, AggValueSlot slot=AggValueSlot::ignored())
LValue emitDeclRefLValue(const clang::DeclRefExpr *e)
llvm::DenseMap< const clang::ValueDecl *, clang::FieldDecl * > lambdaCaptureFields
mlir::Value emitAlignmentAssumption(mlir::Value ptrValue, QualType ty, SourceLocation loc, SourceLocation assumptionLoc, int64_t alignment, mlir::Value offsetValue=nullptr)
LValue makeAddrLValue(Address addr, QualType ty, AlignmentSource source=AlignmentSource::Type)
LValue emitPredefinedLValue(const PredefinedExpr *e)
RValue emitAnyExpr(const clang::Expr *e, AggValueSlot aggSlot=AggValueSlot::ignored(), bool ignoreResult=false)
Emit code to compute the specified expression which can have any type.
void emitCXXDestructorCall(const CXXDestructorDecl *dd, CXXDtorType type, bool forVirtualBase, bool delegating, Address thisAddr, QualType thisTy)
void emitLambdaStaticInvokeBody(const CXXMethodDecl *md)
void instantiateIndirectGotoBlock()
CIRGenFunction(CIRGenModule &cgm, CIRGenBuilderTy &builder, bool suppressNewContext=false)
std::optional< mlir::Location > currSrcLoc
Use to track source locations across nested visitor traversals.
LValue emitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *e)
LValue emitExtVectorElementExpr(const ExtVectorElementExpr *e)
clang::ASTContext & getContext() const
void setAddrOfLocalVar(const clang::VarDecl *vd, Address addr)
Set the address of a local variable.
mlir::Value cxxabiThisValue
mlir::LogicalResult emitStmt(const clang::Stmt *s, bool useCurrentScope, llvm::ArrayRef< const Attr * > attrs={})
Address emitVAListRef(const Expr *e)
Build a "reference" to a va_list; this is either the address or the value of the expression,...
mlir::LogicalResult emitCompoundStmtWithoutScope(const clang::CompoundStmt &s, Address *lastValue=nullptr, AggValueSlot slot=AggValueSlot::ignored())
void emitIgnoredExpr(const clang::Expr *e)
Emit code to compute the specified expression, ignoring the result.
Address createMemTemp(QualType t, mlir::Location loc, const Twine &name="tmp", Address *alloca=nullptr, mlir::OpBuilder::InsertPoint ip={})
Create a temporary memory object of the given type, with appropriate alignmen and cast it to the defa...
LValue emitCompoundLiteralLValue(const CompoundLiteralExpr *e)
This class organizes the cross-function state that is used while generating CIR code.
DiagnosticBuilder errorNYI(SourceLocation, llvm::StringRef)
Helpers to emit "not yet implemented" error diagnostics.
bool inheritingCtorHasParams(const InheritedConstructor &inherited, CXXCtorType type)
Determine if a C++ inheriting constructor should have parameters matching those of its inherited cons...
Type for representing both the decl and type of parameters to a function.
Address getAddress() const
static LValue makeAddr(Address address, clang::QualType t, LValueBaseInfo baseInfo)
clang::QualType getType() const
mlir::Value getPointer() const
LValueBaseInfo getBaseInfo() const
This trivial value class is used to represent the result of an expression that is evaluated.
mlir::Value getValue() const
Return the value of this scalar value.
Represents a C++ constructor within a class.
Represents a C++ destructor within a class.
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
QualType getFunctionObjectParameterType() const
bool isAbstract() const
Determine whether this class has a pure virtual function.
bool isEmpty() const
Determine whether this is an empty class in the sense of (C++11 [meta.unary.prop]).
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
CastKind getCastKind() const
CharUnits - This is an opaque type for sizes expressed in character units.
bool isZero() const
isZero - Test whether the quantity equals zero.
CompoundStmt - This represents a group of statements like { stmt stmt }.
bool isImplicit() const
isImplicit - Indicates whether the declaration was implicitly generated by the implementation.
Decl * getNonClosureContext()
Find the innermost non-closure ancestor of this declaration, walking up through blocks,...
SourceLocation getLocation() const
This represents one expression.
bool EvaluateAsInt(EvalResult &Result, const ASTContext &Ctx, SideEffectsKind AllowSideEffects=SE_NoSideEffects, bool InConstantContext=false) const
EvaluateAsInt - Return true if this is a constant which we can fold and convert to an integer,...
FPOptions getFPFeaturesInEffect(const LangOptions &LO) const
Returns the set of floating point options that apply to this expression.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
LangOptions::FPExceptionModeKind getExceptionMode() const
RoundingMode getRoundingMode() const
Represents a function declaration or definition.
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
FunctionDecl * getPreviousDecl()
Return the previous declaration of this declaration or NULL if this is the first declaration.
GlobalDecl - represents a global declaration.
CXXCtorType getCtorType() const
const Decl * getDecl() const
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
static ImplicitParamDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, ImplicitParamKind ParamKind)
Create implicit parameter.
Describes an C or C++ initializer list.
bool isTransparent() const
Is this a transparent initializer list (that is, an InitListExpr that is purely syntactic,...
const Expr * getInit(unsigned Init) const
FPExceptionModeKind
Possible floating point exception behavior.
@ FPE_Default
Used internally to represent initial unspecified value.
@ FPE_Strict
Strictly preserve the floating-point exception semantics.
@ FPE_MayTrap
Transformations do not cause new exceptions but may hide some.
@ FPE_Ignore
Assume that floating-point exceptions are masked.
Represents an unpacked "presumed" location which can be presented to the user.
unsigned getColumn() const
Return the presumed column number of this location.
const char * getFilename() const
Return the presumed filename of this location.
unsigned getLine() const
Return the presumed line number of this location.
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
Expr * getResultExpr()
Return the result-bearing expression, or null if there is none.
ArrayRef< Expr * > semantics()
A (possibly-)qualified type.
bool isTriviallyCopyableType(const ASTContext &Context) const
Return true if this is a trivially copyable type (C++0x [basic.types]p9)
QualType getUnqualifiedType() const
Retrieve the unqualified variant of the given type, removing as little sugar as possible.
Encodes a location in the source.
bool isValid() const
Return true if this is a valid SourceLocation object.
This class handles loading and caching of source files into memory.
PresumedLoc getPresumedLoc(SourceLocation Loc, bool UseLineDirectives=true) const
Returns the "presumed" location of a SourceLocation specifies.
A trivial tuple used to represent a source range.
SourceLocation getEnd() const
SourceLocation getBegin() const
Stmt - This represents one statement.
StmtClass getStmtClass() const
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
const char * getStmtClassName() const
SourceLocation getBeginLoc() const LLVM_READONLY
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
bool isAnyComplexType() const
bool isVariablyModifiedType() const
Whether this type is a variably-modified type (C99 6.7.5).
TypeClass getTypeClass() const
const T * getAs() const
Member-template getAs<specific type>'.
Represents a variable declaration or definition.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
Represents a C array with a specified size that is not an integer-constant-expression.
Expr * getSizeExpr() const
@ Type
The l-value was considered opaque, so the alignment was determined from a type.
@ Decl
The l-value was an access to a declared entity or something equivalently strong, like the address of ...
static bool previousOpIsNonYieldingCleanup(mlir::Block *block)
static std::string getVersionedTmpName(llvm::StringRef name, unsigned cnt)
static bool mayDropFunctionReturn(const ASTContext &astContext, QualType returnType)
static mlir::Value emitArgumentDemotion(CIRGenFunction &cgf, const VarDecl *var, mlir::Value value)
An argument came in as a promoted argument; demote it back to its declared type.
static void eraseEmptyAndUnusedBlocks(cir::FuncOp func)
static std::variant< LValue, RValue > emitPseudoObjectExpr(CIRGenFunction &cgf, const PseudoObjectExpr *e, bool forLValue, AggValueSlot slot)
static llvm::fp::ExceptionBehavior toConstrainedExceptMd(LangOptions::FPExceptionModeKind kind)
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const AstTypeMatcher< ArrayType > arrayType
const internal::VariadicDynCastAllOfMatcher< Stmt, Expr > expr
Matches expressions.
CXXCtorType
C++ constructor types.
@ Ctor_Base
Base object ctor.
@ Ctor_Complete
Complete object ctor.
bool isa(CodeGen::Address addr)
CXXDtorType
C++ destructor types.
@ Dtor_VectorDeleting
Vector deleting dtor.
@ Dtor_Comdat
The COMDAT used for dtors.
@ Dtor_Unified
GCC-style unified dtor.
@ Dtor_Base
Base object dtor.
@ Dtor_Complete
Complete object dtor.
@ Dtor_Deleting
Deleting dtor.
U cast(CodeGen::Address addr)
@ Other
Other implicit parameter.
static bool fastMathFuncAttributes()
static bool vtableInitialization()
static bool constructABIArgDirectExtend()
static bool runCleanupsScope()
static bool emitTypeCheck()
static bool fastMathGuard()
static bool fastMathFlags()
static bool generateDebugInfo()
static bool incrementProfileCounter()
Represents a scope, including function bodies, compound statements, and the substatements of if/while...
llvm::ArrayRef< mlir::Block * > getRetBlocks()
LexicalScope(CIRGenFunction &cgf, mlir::Location loc, mlir::Block *eb)
cir::TryOp getClosestTryParent()
mlir::Location getRetLoc(mlir::Block *b)
EvalResult is a struct with detailed info about an evaluated expression.
APValue Val
Val - This is the value the expression can be folded to.