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;
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())
281 bool isEntryBlock = builder.getInsertionBlock()->isEntryBlock();
282 if (!isEntryBlock && curBlock->empty()) {
284 for (mlir::Block *retBlock : retBlocks) {
285 if (retBlock->getUses().empty())
291 if (localScope->
depth == 0) {
295 mlir::Location retLoc = localScope->
getRetLoc(retBlock);
296 if (retBlock->getUses().empty()) {
299 cir::BrOp::create(builder, retLoc, retBlock);
303 emitImplicitReturn();
310 if (!localScope->
isTernary() && !curBlock->mightHaveTerminator()) {
311 !retVal ? cir::YieldOp::create(builder, localScope->endLoc)
312 : cir::YieldOp::create(builder, localScope->endLoc, retVal);
316cir::ReturnOp CIRGenFunction::LexicalScope::emitReturn(mlir::Location loc) {
319 auto fn = dyn_cast<cir::FuncOp>(cgf.curFn);
320 assert(fn &&
"emitReturn from non-function");
322 if (!fn.getFunctionType().hasVoidReturn()) {
324 auto value = cir::LoadOp::create(
325 builder, loc, fn.getFunctionType().getReturnType(), *cgf.fnRetAlloca);
326 return cir::ReturnOp::create(builder, loc,
329 return cir::ReturnOp::create(builder, loc);
339 return classDecl->hasTrivialDestructor();
346 mlir::Operation *op = &block->back();
347 auto cleanupScopeOp = mlir::dyn_cast<cir::CleanupScopeOp>(op);
355 for (mlir::Block &bodyBlock : cleanupScopeOp.getBodyRegion()) {
356 if (bodyBlock.mightHaveTerminator()) {
357 if (mlir::isa<cir::YieldOp>(bodyBlock.getTerminator()))
359 assert(!mlir::isa<cir::BreakOp>(bodyBlock.getTerminator()) &&
360 !mlir::isa<cir::ContinueOp>(bodyBlock.getTerminator()) &&
361 !mlir::isa<cir::ResumeOp>(bodyBlock.getTerminator()));
367void CIRGenFunction::LexicalScope::emitImplicitReturn() {
368 CIRGenBuilderTy &builder = cgf.getBuilder();
369 LexicalScope *localScope = cgf.curLexScope;
378 if (cgf.getLangOpts().CPlusPlus && !fd->hasImplicitReturnZero() &&
379 !cgf.sawAsmBlock && !fd->getReturnType()->isVoidType() &&
380 builder.getInsertionBlock() &&
382 bool shouldEmitUnreachable =
383 cgf.cgm.getCodeGenOpts().StrictReturn ||
386 if (shouldEmitUnreachable) {
388 if (cgf.cgm.getCodeGenOpts().OptimizationLevel == 0)
389 cir::TrapOp::create(builder, localScope->endLoc);
391 cir::UnreachableOp::create(builder, localScope->endLoc);
392 builder.clearInsertionPoint();
397 (void)emitReturn(localScope->endLoc);
405 scope = scope->parentScope;
418 if (value.getType() == ty)
421 assert((mlir::isa<cir::IntType>(ty) || cir::isAnyFloatingPointType(ty)) &&
422 "unexpected promotion type");
424 if (mlir::isa<cir::IntType>(ty))
425 return cgf.
getBuilder().CIRBaseBuilderTy::createIntCast(value, ty);
431 mlir::Block *entryBB,
435 if (fd && fd->
hasAttr<NakedAttr>()) {
436 cgm.errorNYI(bodyBeginLoc,
"naked function decl");
440 for (
const auto nameValue : llvm::zip(args, entryBB->getArguments())) {
441 const VarDecl *paramVar = std::get<0>(nameValue);
442 mlir::Value paramVal = std::get<1>(nameValue);
445 paramVal.setLoc(paramLoc);
447 mlir::Value addrVal =
452 declare(addrVal, paramVar, paramVar->
getType(), paramLoc, alignment,
465 mlir::Location fnBodyBegin =
getLoc(bodyBeginLoc);
466 builder.CIRBaseBuilderTy::createStore(fnBodyBegin, paramVal, addrVal);
468 assert(builder.getInsertionBlock() &&
"Should be valid");
472 cir::FuncOp fn, cir::FuncType funcType,
476 "CIRGenFunction can only be used for one function at a time");
484 const auto *fd = dyn_cast_or_null<FunctionDecl>(d);
489 mlir::Block *entryBB = &fn.getBlocks().front();
490 builder.setInsertionPointToStart(entryBB);
496 if (
Stmt *body = fd->getBody())
497 bodyBeginLoc = body->getBeginLoc();
499 bodyBeginLoc = fd->getLocation();
511 if (
Stmt *body = fd->getBody())
512 bodyEndLoc = body->getEndLoc();
514 bodyEndLoc = fd->getLocation();
516 emitAndUpdateRetAlloca(returnType,
getLoc(bodyEndLoc),
517 getContext().getTypeAlignInChars(returnType));
526 if (fd && fd->hasImplicitReturnZero()) {
528 mlir::Location bodyBeginMLIRLoc =
getLoc(bodyBeginLoc);
529 mlir::Value zero = builder.getNullValue(cirRetTy, bodyBeginMLIRLoc);
530 builder.CIRBaseBuilderTy::createStore(bodyBeginMLIRLoc, zero,
539 if (isa_and_nonnull<CXXMethodDecl>(d) &&
541 cgm.getCXXABI().emitInstanceFunctionProlog(loc, *
this);
544 if (md->getParent()->isLambda() && md->getOverloadedOperator() == OO_Call) {
546 auto fn = dyn_cast<cir::FuncOp>(
curFn);
547 assert(fn &&
"lambda in non-function region");
573 for (
auto *fd : md->getParent()->fields()) {
574 if (fd->hasCapturedVLAType())
575 cgm.errorNYI(loc,
"lambda captured VLA type");
591 if (!fd || !fd->hasAttr<NakedAttr>()) {
592 for (
const VarDecl *vd : args) {
597 if (
const auto *pvd = dyn_cast<ParmVarDecl>(vd))
598 ty = pvd->getOriginalType();
608 for (cir::BlockAddressOp &blockAddress :
cgm.unresolvedBlockAddressToLabel) {
609 cir::LabelOp labelOp =
610 cgm.lookupBlockAddressInfo(blockAddress.getBlockAddrInfo());
611 assert(labelOp &&
"expected cir.labelOp to already be emitted");
612 cgm.updateResolvedBlockAddress(blockAddress, labelOp);
614 cgm.unresolvedBlockAddressToLabel.clear();
622 mlir::OpBuilder::InsertionGuard guard(builder);
624 for (
auto &[blockAdd, labelOp] :
cgm.blockAddressToLabel) {
625 succesors.push_back(labelOp->getBlock());
626 rangeOperands.push_back(labelOp->getBlock()->getArguments());
628 cir::IndirectBrOp::create(builder, builder.getUnknownLoc(),
630 rangeOperands, succesors);
631 cgm.blockAddressToLabel.clear();
646 indrBr.setPoison(
true);
662 "deferred conditional cleanups were not consumed by a "
663 "FullExprCleanupScope");
670 if (
const CompoundStmt *block = dyn_cast<CompoundStmt>(body))
681 for (mlir::Block &block : func.getBlocks()) {
682 if (block.empty() && block.getUses().empty())
683 blocksToDelete.push_back(&block);
685 for (mlir::Block *block : blocksToDelete)
690 cir::FuncType funcType) {
694 if (funcDecl->isInlineBuiltinDeclaration()) {
698 std::string fdInlineName = (
cgm.getMangledName(funcDecl) +
".inline").str();
700 mlir::cast_or_null<cir::FuncOp>(
cgm.getGlobalValue(fdInlineName));
702 mlir::OpBuilder::InsertionGuard guard(builder);
703 builder.setInsertionPoint(fn);
704 clone = cir::FuncOp::create(builder, fn.getLoc(), fdInlineName,
705 fn.getFunctionType());
706 cgm.insertGlobalSymbol(clone);
707 clone.setLinkage(cir::GlobalLinkageKind::InternalLinkage);
708 clone.setSymVisibility(
"private");
709 clone.setInlineKind(cir::InlineKind::AlwaysInline);
711 fn.setLinkage(cir::GlobalLinkageKind::ExternalLinkage);
712 fn.setSymVisibility(
"private");
720 if (LLVM_UNLIKELY(pd->isInlineBuiltinDeclaration())) {
721 std::string inlineName = funcDecl->getName().str() +
".inline";
722 if (
auto inlineFn = mlir::cast_or_null<cir::FuncOp>(
723 cgm.getGlobalValue(inlineName))) {
728 .replaceAllSymbolUses(fn.getSymNameAttr(),
cgm.getModule())
730 llvm_unreachable(
"Failed to replace inline builtin symbol uses");
731 cgm.eraseGlobalSymbol(inlineFn);
740 Stmt *body = funcDecl->getBody();
745 : builder.getUnknownLoc()};
748 return clangLoc.isValid() ?
getLoc(clangLoc) : builder.getUnknownLoc();
750 const mlir::Location fusedLoc = mlir::FusedLoc::get(
752 {validMLIRLoc(bodyRange.
getBegin()), validMLIRLoc(bodyRange.
getEnd())});
753 mlir::Block *entryBB = fn.addEntryBlock();
765 if (funcDecl->UsesFPIntrin() || funcDecl->hasAttr<StrictFPAttr>()) {
766 cgm.errorNYI(loc,
"STDC FENV_ACCESS");
771 if (body && isa_and_nonnull<CoroutineBodyStmt>(body))
772 llvm::append_range(
fnArgs, funcDecl->parameters());
779 funcDecl->hasAttr<CUDAGlobalAttr>()) {
780 cgm.getCUDARuntime().emitDeviceStub(*
this, fn, args);
800 llvm_unreachable(
"no definition for normal function");
803 if (mlir::failed(fn.verifyBody()))
818 assert((
cgm.getTarget().getCXXABI().hasConstructorVariants() ||
820 "can only generate complete ctor for this ABI");
825 cgm.getTarget().getCXXABI().hasConstructorVariants()) {
831 Stmt *body = ctor->getBody(definition);
832 assert(definition == ctor &&
"emitting wrong constructor body");
834 bool isTryBody = isa_and_nonnull<CXXTryStmt>(body);
843 Stmt *emitterBody =
nullptr;
846 : ctor(ctor), ctorType(ctorType), args(args),
848 ~ctorTryBodyEmitter()
override =
default;
858 return cgf.
emitStmt(emitterBody,
true);
862 ctorTryBodyEmitter emitter{ctor, ctorType, args, isTryBody, body};
863 mlir::LogicalResult bodyRes =
869 if (bodyRes.failed())
870 cgm.errorNYI(ctor->getSourceRange(),
871 "emitConstructorBody: emit body statement failed.");
915 const bool isTryBody = isa_and_nonnull<CXXTryStmt>(body);
931 llvm_unreachable(
"not expecting a unified dtor");
933 llvm_unreachable(
"not expecting a COMDAT");
936 llvm_unreachable(
"already handled deleting case");
939 assert((body ||
getTarget().getCXXABI().isMicrosoft()) &&
940 "can't emit a dtor without a body for non-Microsoft ABIs");
968 assert(dtor->
isImplicit() &&
"bodyless dtor not implicit");
994 CharUnits align =
cgm.getNaturalTypeAlignment(ty, &baseInfo);
1001 CharUnits alignment =
cgm.getNaturalTypeAlignment(ty, &baseInfo);
1009static llvm::fp::ExceptionBehavior
1013 return llvm::fp::ebIgnore;
1015 return llvm::fp::ebMayTrap;
1017 return llvm::fp::ebStrict;
1019 llvm_unreachable(
"expected explicitly initialized exception behavior");
1021 llvm_unreachable(
"unsupported FP exception behavior");
1027 QualType retTy = fd->getReturnType();
1032 const auto *md = dyn_cast<CXXMethodDecl>(fd);
1033 if (md && md->isImplicitObjectMemberFunction()) {
1034 if (
cgm.getCXXABI().hasThisReturn(gd))
1035 cgm.errorNYI(fd->getSourceRange(),
"this return");
1036 else if (
cgm.getCXXABI().hasMostDerivedReturn(gd))
1037 cgm.errorNYI(fd->getSourceRange(),
"most derived return");
1038 cgm.getCXXABI().buildThisParam(*
this, args);
1041 bool passedParams =
true;
1042 if (
const auto *cd = dyn_cast<CXXConstructorDecl>(fd))
1043 if (
auto inherited = cd->getInheritedConstructor())
1048 for (
auto *param : fd->parameters()) {
1049 args.push_back(param);
1050 if (!param->hasAttr<PassObjectSizeAttr>())
1054 getContext(), param->getDeclContext(), param->getLocation(),
1057 args.push_back(implicit);
1062 cgm.getCXXABI().addImplicitStructorParams(*
this, retTy, args);
1073 assert(e->
isTransparent() &&
"non-transparent glvalue init list");
1077static std::variant<LValue, RValue>
1082 llvm::scope_exit opaque_cleanup{
1083 [&]() { llvm::for_each(opaques, [&](OVMD &o) { o.unbind(cgf); }); }};
1087 std::variant<LValue, RValue> result;
1092 if (
const auto *ov = dyn_cast<OpaqueValueExpr>(semantic)) {
1095 if (ov->isUnique()) {
1096 assert(ov != resultExpr &&
1097 "A unique OVE cannot be used as the result expression");
1104 if (ov == resultExpr && ov->
isPRValue() && !forLValue &&
1107 "emitPseudoObjectExpr for RValue & aggregate kind");
1109 opaqueData = OVMD::bind(cgf, ov, ov->getSourceExpr());
1112 if (ov == resultExpr) {
1116 "emitPseudoObjectExpr as result");
1121 "emitPseudoObjectExpr as an RValue");
1124 opaques.push_back(opaqueData);
1125 }
else if (semantic == resultExpr) {
1136 "emitPseudoObjectExpr as an ignored value");
1147 return std::get<RValue>(
1164 "emitLValue: unsupported l-value class");
1167 case Expr::ObjCPropertyRefExprClass:
1168 llvm_unreachable(
"cannot emit a property reference directly");
1170 case Expr::ObjCSelectorExprClass:
1172 "emitLValue: ObjCSelectorExpr");
1174 case Expr::ObjCIsaExprClass:
1177 case Expr::BinaryOperatorClass:
1179 case Expr::CompoundAssignOperatorClass: {
1182 ty = at->getValueType();
1188 case Expr::CallExprClass:
1189 case Expr::CXXMemberCallExprClass:
1190 case Expr::CXXOperatorCallExprClass:
1191 case Expr::UserDefinedLiteralClass:
1193 case Expr::CXXRewrittenBinaryOperatorClass:
1195 "emitLValue: CXXRewrittenBinaryOperator");
1197 case Expr::VAArgExprClass:
1200 case Expr::DeclRefExprClass:
1202 case Expr::ConstantExprClass:
1205 case Expr::ParenExprClass:
1207 case Expr::GenericSelectionExprClass:
1209 case Expr::PredefinedExprClass:
1211 case Expr::StringLiteralClass:
1213 case Expr::ObjCEncodeExprClass:
1215 "emitLValue: ObjCEncodeExpr");
1217 case Expr::PseudoObjectExprClass:
1219 case Expr::InitListExprClass:
1221 case Expr::CXXTemporaryObjectExprClass:
1222 case Expr::CXXConstructExprClass:
1224 case Expr::CXXBindTemporaryExprClass:
1226 case Expr::CXXUuidofExprClass:
1228 "emitLValue: CXXUuidofExpr");
1230 case Expr::LambdaExprClass:
1233 case Expr::ExprWithCleanupsClass: {
1250 case Expr::CXXDefaultArgExprClass: {
1255 case Expr::CXXDefaultInitExprClass: {
1260 case Expr::CXXTypeidExprClass:
1262 case Expr::ObjCMessageExprClass:
1264 "emitLValue: ObjCMessageExpr");
1266 case Expr::ObjCIvarRefExprClass:
1268 "emitLValue: ObjCIvarRefExpr");
1270 case Expr::StmtExprClass:
1273 case Expr::UnaryOperatorClass:
1275 case Expr::ArraySubscriptExprClass:
1277 case Expr::MatrixSingleSubscriptExprClass:
1279 "emitLValue: MatrixSingleSubscriptExpr");
1281 case Expr::MatrixSubscriptExprClass:
1283 "emitLValue: MatrixSubscriptExpr");
1285 case Expr::ArraySectionExprClass:
1287 "emitLValue: ArraySectionExpr");
1289 case Expr::ExtVectorElementExprClass:
1291 case Expr::MatrixElementExprClass:
1293 "emitLValue: MatrixElementExpr");
1295 case Expr::CXXThisExprClass:
1298 case Expr::MemberExprClass:
1300 case Expr::CompoundLiteralExprClass:
1302 case Expr::ConditionalOperatorClass:
1304 case Expr::BinaryConditionalOperatorClass:
1306 case Expr::ChooseExprClass:
1308 case Expr::OpaqueValueExprClass:
1310 case Expr::SubstNonTypeTemplateParmExprClass:
1312 case Expr::ImplicitCastExprClass:
1313 case Expr::CStyleCastExprClass:
1314 case Expr::CXXFunctionalCastExprClass:
1315 case Expr::CXXStaticCastExprClass:
1316 case Expr::CXXDynamicCastExprClass:
1317 case Expr::CXXReinterpretCastExprClass:
1318 case Expr::CXXConstCastExprClass:
1320 case Expr::CXXAddrspaceCastExprClass:
1321 case Expr::ObjCBridgedCastExprClass:
1325 "emitLValue: addrspace or ObjC bridged cast");
1327 case Expr::MaterializeTemporaryExprClass:
1329 case Expr::CoawaitExprClass:
1332 case Expr::CoyieldExprClass:
1335 case Expr::PackIndexingExprClass:
1337 "emitLValue: PackIndexingExpr");
1339 case Expr::HLSLOutArgExprClass:
1340 llvm_unreachable(
"cannot emit a HLSL out argument directly");
1346 llvm::raw_svector_ostream
out(buffer);
1348 return std::string(
out.str());
1368 cgm.errorNYI(loc,
"Cast the dest ptr to the appropriate i8 pointer type");
1377 "emitNullInitialization for zero size VariableArrayType");
1387 if (!
cgm.getTypes().isZeroInitializable(ty)) {
1388 cgm.errorNYI(loc,
"type is not zero initializable");
1395 const mlir::Value zeroValue = builder.getNullValue(
convertType(ty), loc);
1396 builder.createStore(loc, zeroValue, destPtr);
1408 ConstructorHelper(fpFeatures);
1411void CIRGenFunction::CIRGenFPOptionsRAII::ConstructorHelper(
1413 oldFPFeatures = cgf.curFPFeatures;
1414 cgf.curFPFeatures = fpFeatures;
1416 oldExcept = cgf.builder.getDefaultConstrainedExcept();
1417 oldRounding = cgf.builder.getDefaultConstrainedRounding();
1419 if (oldFPFeatures == fpFeatures)
1425 [[maybe_unused]] llvm::RoundingMode newRoundingBehavior =
1428 [[maybe_unused]] llvm::fp::ExceptionBehavior newExceptionBehavior =
1436 assert((cgf.curFuncDecl ==
nullptr || cgf.builder.getIsFPConstrained() ||
1439 (newExceptionBehavior == llvm::fp::ebIgnore &&
1440 newRoundingBehavior == llvm::RoundingMode::NearestTiesToEven)) &&
1441 "FPConstrained should be enabled on entire function");
1448 cgf.curFPFeatures = oldFPFeatures;
1449 cgf.builder.setDefaultConstrainedExcept(oldExcept);
1450 cgf.builder.setDefaultConstrainedRounding(oldRounding);
1457 if (ce->
getCastKind() == CK_UncheckedDerivedToBase)
1467 if (ice->isGLValue())
1483 mlir::Value numVLAElements =
nullptr;
1495 baseType = elementType;
1496 return numVLAElements;
1509 uint64_t countFromCLAs = 1;
1512 auto cirArrayType = mlir::dyn_cast<cir::ArrayType>(addr.
getElementType());
1514 while (cirArrayType) {
1516 countFromCLAs *= cirArrayType.getSize();
1520 mlir::dyn_cast<cir::ArrayType>(cirArrayType.getElementType());
1524 "CIR and Clang types are out-of-sync");
1531 cgm.errorNYI(*
currSrcLoc,
"length for non-array underlying types");
1536 mlir::Value numElements =
1542 builder.createMul(numVLAElements.getLoc(), numVLAElements, numElements,
1553 mlir::OpBuilder::InsertionGuard guard(builder);
1555 builder.createBlock(builder.getBlock()->getParent(), {}, {voidPtrTy},
1556 {builder.getUnknownLoc()});
1561 SourceLocation assumptionLoc, int64_t alignment, mlir::Value offsetValue) {
1563 mlir::Location assumeLoc =
getLoc(assumptionLoc);
1564 mlir::Value alignValue = builder.getUInt64(alignment, assumeLoc);
1565 mlir::Value cond = builder.getBool(
true, assumeLoc);
1568 bundleArgs.push_back(offsetValue);
1569 cir::AssumeOp::create(builder, assumeLoc, cond, cir::AssumeBundleKind::Align,
1576 int64_t alignment, mlir::Value offsetValue) {
1585 cgm.getASTContext().getAsVariableArrayType(
type);
1586 assert(vla &&
"type was not a variable array type!");
1593 mlir::Value numElements;
1597 elementType =
type->getElementType();
1599 assert(vlaSize &&
"no size for VLA!");
1600 assert(vlaSize.getType() ==
sizeTy);
1603 numElements = vlaSize;
1609 builder.createMul(numElements.getLoc(), numElements, vlaSize,
1612 }
while ((
type =
getContext().getAsVariableArrayType(elementType)));
1614 assert(numElements &&
"Undefined elements number");
1615 return {numElements, elementType};
1621 assert(vlaSize &&
"no size for VLA!");
1622 assert(vlaSize.getType() ==
sizeTy);
1629 assert(
type->isVariablyModifiedType() &&
1630 "Must pass variably modified type to EmitVLASizes!");
1635 assert(
type->isVariablyModifiedType());
1637 const Type *ty =
type.getTypePtr();
1639 case Type::CountAttributed:
1640 case Type::PackIndexing:
1641 case Type::ArrayParameter:
1642 case Type::HLSLAttributedResource:
1643 case Type::HLSLInlineSpirv:
1644 case Type::PredefinedSugar:
1645 cgm.errorNYI(
"CIRGenFunction::emitVariablyModifiedType");
1648#define TYPE(Class, Base)
1649#define ABSTRACT_TYPE(Class, Base)
1650#define NON_CANONICAL_TYPE(Class, Base)
1651#define DEPENDENT_TYPE(Class, Base) case Type::Class:
1652#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base)
1653#include "clang/AST/TypeNodes.inc"
1655 "dependent type must be resolved before the CIR codegen");
1661 case Type::ExtVector:
1662 case Type::ConstantMatrix:
1666 case Type::TemplateSpecialization:
1667 case Type::ObjCTypeParam:
1668 case Type::ObjCObject:
1669 case Type::ObjCInterface:
1670 case Type::ObjCObjectPointer:
1672 case Type::OverflowBehavior:
1673 llvm_unreachable(
"type class is never variably-modified!");
1675 case Type::Adjusted:
1687 case Type::BlockPointer:
1691 case Type::LValueReference:
1692 case Type::RValueReference:
1696 case Type::MemberPointer:
1700 case Type::ConstantArray:
1701 case Type::IncompleteArray:
1706 case Type::VariableArray: {
1723 entry = builder.createBoolIntToIntCast(size,
sizeTy);
1730 case Type::FunctionProto:
1731 case Type::FunctionNoProto:
1737 case Type::UnaryTransform:
1738 case Type::Attributed:
1739 case Type::BTFTagAttributed:
1740 case Type::SubstTemplateTypeParm:
1741 case Type::MacroQualified:
1747 case Type::Decltype:
1749 case Type::DeducedTemplateSpecialization:
1753 case Type::TypeOfExpr:
1766 }
while (
type->isVariablyModifiedType());
1770 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
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 emitCXXBindTemporaryLValue(const CXXBindTemporaryExpr *e)
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 emitCXXConstructLValue(const CXXConstructExpr *e)
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.
child_iterator child_begin()
StmtClass getStmtClass() const
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
child_iterator child_end()
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.