19#include "mlir/Dialect/Ptr/IR/MemorySpaceInterfaces.h"
20#include "mlir/IR/BuiltinAttributes.h"
21#include "mlir/IR/Value.h"
69 llvm::StringRef fieldName,
70 unsigned fieldIndex) {
89 auto fieldPtr = cir::PointerType::get(fieldType);
90 bool needsBitcast =
false;
93 mlir::Type memberType = layout.
getCIRType().getMembers()[idx];
94 fieldPtr = cir::PointerType::get(memberType);
101 mlir::Value addr = builder.createGetMember(loc, fieldPtr, base.
getPointer(),
102 fieldName, fieldIndex);
108 addr = builder.createPtrBitcast(addr, fieldType);
111 layout.
getCIRType().getElementOffset(
cgm.getDataLayout().layout, idx));
120 assert(
expr->getType()->isPointerType() ||
121 expr->getType()->isObjCObjectPointerType());
125 if (
auto const *ce = dyn_cast<CastExpr>(
expr)) {
126 if (
const auto *ece = dyn_cast<ExplicitCastExpr>(ce))
127 cgm.emitExplicitCastExprType(ece);
129 switch (ce->getCastKind()) {
133 case CK_AddressSpaceConversion: {
134 if (
const auto *ptrTy =
135 ce->getSubExpr()->getType()->getAs<
PointerType>()) {
136 if (ptrTy->getPointeeType()->isVoidType())
144 *baseInfo = innerBaseInfo;
149 const QualType pointeeType =
expr->getType()->getPointeeType();
151 cgm.getNaturalTypeAlignment(pointeeType, &targetTypeBaseInfo);
164 const mlir::Type eltTy =
176 case CK_ArrayToPointerDecay:
179 case CK_UncheckedDerivedToBase:
180 case CK_DerivedToBase: {
185 ce->getSubExpr()->getType()->getPointeeCXXRecordDecl();
191 case CK_AnyPointerToBlockPointerCast:
192 case CK_BaseToDerived:
193 case CK_BaseToDerivedMemberPointer:
194 case CK_BlockPointerToObjCPointerCast:
195 case CK_BuiltinFnToFnPtr:
196 case CK_CPointerToObjCPointerCast:
197 case CK_DerivedToBaseMemberPointer:
199 case CK_FunctionToPointerDecay:
200 case CK_IntegralToPointer:
201 case CK_LValueToRValue:
202 case CK_LValueToRValueBitCast:
203 case CK_NullToMemberPointer:
204 case CK_NullToPointer:
205 case CK_ReinterpretMemberPointer:
206 case CK_UserDefinedConversion:
212 case CK_ARCConsumeObject:
213 case CK_ARCExtendBlockObject:
214 case CK_ARCProduceObject:
215 case CK_ARCReclaimReturnedObject:
216 case CK_AtomicToNonAtomic:
217 case CK_BooleanToSignedIntegral:
218 case CK_ConstructorConversion:
219 case CK_CopyAndAutoreleaseBlockObject:
221 case CK_FixedPointCast:
222 case CK_FixedPointToBoolean:
223 case CK_FixedPointToFloating:
224 case CK_FixedPointToIntegral:
225 case CK_FloatingCast:
226 case CK_FloatingComplexCast:
227 case CK_FloatingComplexToBoolean:
228 case CK_FloatingComplexToIntegralComplex:
229 case CK_FloatingComplexToReal:
230 case CK_FloatingRealToComplex:
231 case CK_FloatingToBoolean:
232 case CK_FloatingToFixedPoint:
233 case CK_FloatingToIntegral:
234 case CK_HLSLAggregateSplatCast:
235 case CK_HLSLArrayRValue:
236 case CK_HLSLElementwiseCast:
237 case CK_HLSLVectorTruncation:
238 case CK_HLSLMatrixTruncation:
239 case CK_IntToOCLSampler:
240 case CK_IntegralCast:
241 case CK_IntegralComplexCast:
242 case CK_IntegralComplexToBoolean:
243 case CK_IntegralComplexToFloatingComplex:
244 case CK_IntegralComplexToReal:
245 case CK_IntegralRealToComplex:
246 case CK_IntegralToBoolean:
247 case CK_IntegralToFixedPoint:
248 case CK_IntegralToFloating:
249 case CK_LValueBitCast:
251 case CK_MemberPointerToBoolean:
252 case CK_NonAtomicToAtomic:
253 case CK_ObjCObjectLValueCast:
254 case CK_PointerToBoolean:
255 case CK_PointerToIntegral:
259 case CK_ZeroToOCLOpaqueType:
264 cgm.errorNYI(ce->getSourceRange(),
265 "unexpected cast for emitPointerWithAlignment: ",
266 ce->getCastKindName());
274 if (uo->getOpcode() == UO_AddrOf) {
284 if (
auto const *call = dyn_cast<CallExpr>(
expr)) {
285 switch (call->getBuiltinCallee()) {
288 case Builtin::BIaddressof:
289 case Builtin::BI__addressof:
290 case Builtin::BI__builtin_addressof: {
308 auto getScalarSizeInBits = [&](mlir::Type ty) ->
unsigned {
309 mlir::Type scalarTy = mlir::isa<cir::VectorType>(ty)
310 ? mlir::cast<cir::VectorType>(ty).getElementType()
316 mlir::Value srcVal = src.
getValue();
319 getScalarSizeInBits(srcVal.getType())) {
322 "emitStoreThroughExtVectorComponentLValue: dstTySize > srcTysize");
328 "emitStoreThroughExtVectorComponentLValue: HLSL");
338 mlir::Value vec = builder.createLoad(loc, dstAddr, dst.
isVolatile());
340 unsigned numSrcElts = vecTy->getNumElements();
342 if (numDstElts == numSrcElts) {
347 for (
unsigned i = 0; i != numDstElts; ++i)
350 vec = builder.createVecShuffle(loc, srcVal, mask);
351 }
else if (numDstElts > numSrcElts) {
357 std::iota(extMask.begin(), extMask.begin() + numSrcElts, 0);
359 mlir::Value extSrcVal = builder.createVecShuffle(loc, srcVal, extMask);
363 std::iota(mask.begin(), mask.begin() + numDstElts, 0);
372 for (
unsigned i = 0; i != numSrcElts; ++i)
375 vec = builder.createVecShuffle(loc, vec, extSrcVal, mask);
378 llvm_unreachable(
"unexpected shorten vector length");
384 cir::ConstantOp elt = builder.getSInt64(inIdx, loc);
385 vec = cir::VecInsertOp::create(builder, loc, vec, srcVal, elt);
398 const mlir::Value vector =
400 const mlir::Value newVector = cir::VecInsertOp::create(
409 assert(dst.
isBitField() &&
"Unknown LValue type");
414 "emitStoreThroughLValue: non-simple lvalue");
420 assert(src.
isScalar() &&
"Can't emit an aggregate store with this method");
431 "emitGlobalVarDeclLValue: thread_local variable");
443 cir::PointerType realPtrTy = cir::PointerType::get(
444 realVarTy, mlir::cast<cir::PointerType>(v.getType()).getAddrSpace());
445 if (realPtrTy != v.getType())
449 Address addr(v, realVarTy, alignment);
463 bool isNontemporal) {
467 if (clangVecTy->isExtVectorBoolType())
469 "emitStoreOfScalar ExtVectorBoolType");
477 if (vecTy.getSize() == 3 && !
getLangOpts().PreserveVec3Type)
479 "emitStoreOfScalar Vec3 & PreserveVec3Type disabled");
493 assert(addr.
getPointer() &&
"expected pointer to exist");
495 if (currVarDecl && srcAlloca) {
496 const VarDecl *vd = currVarDecl;
497 assert(vd &&
"VarDecl expected");
502 assert(
currSrcLoc &&
"must pass in source location");
503 builder.createStore(*
currSrcLoc, value, addr, isVolatile);
506 cgm.errorNYI(addr.
getPointer().getLoc(),
"emitStoreOfScalar nontemporal");
516 return targetInfo.
getABI().starts_with(
"aapcs");
526 bool useVoaltile =
cgm.getCodeGenOpts().AAPCSBitfieldWidth &&
530 assert(
currSrcLoc &&
"must pass in source location");
532 return builder.createSetBitfield(*
currSrcLoc, resLTy, ptr,
556 mlir::Type fieldType,
559 cir::PointerType fieldPtr = cir::PointerType::get(fieldType);
565 rec.getElementOffset(
cgm.getDataLayout().layout,
index));
573 cgm.getTypes().getCIRGenRecordLayout(field->
getParent());
583 addr = builder.createElementBitCast(loc, addr, info.
storageType);
606 if (
auto *classDecl = dyn_cast<CXXRecordDecl>(rec)) {
607 if (
cgm.getCodeGenOpts().StrictVTablePointers &&
608 classDecl->isDynamicClass()) {
610 "emitLValueForField: strict vtable for dynamic class");
616 llvm::StringRef fieldName = field->
getName();
618 if (
cgm.lambdaFieldToName.count(field))
619 fieldName =
cgm.lambdaFieldToName[field];
634 cgm.getTypes().getCIRGenRecordLayout(field->
getParent());
655 if (field->
hasAttr<AnnotateAttr>()) {
666 "emitLValueForField: __weak attribute");
685 cgm.getTypes().getCIRGenRecordLayout(field->
getParent());
708 ty = atomicTy->getValueType();
711 cgm.errorNYI(
"emitToMemory: extVectorBoolType");
722 ty = atomicTy->getValueType();
725 cgm.errorNYI(
"emitFromMemory: PackedVectorBoolType");
734 assert(0 &&
"NYI: emitStoreOfScalar constant matrix type");
751 if (clangVecTy->isExtVectorBoolType()) {
752 cgm.errorNYI(loc,
"emitLoadOfScalar: ExtVectorBoolType");
760 if (vecTy.getSize() == 3 && !
getLangOpts().PreserveVec3Type)
762 "emitLoadOfScalar Vec3 & PreserveVec3Type disabled");
770 if (mlir::isa<cir::VoidType>(eltTy))
771 cgm.errorNYI(loc,
"emitLoadOfScalar: void type");
775 mlir::Value loadOp = builder.createLoad(
getLoc(loc), addr, isVolatile);
777 cgm.errorNYI(
"emitLoadOfScalar: boolean type with boolean representation");
804 const mlir::Value load =
813 cgm.errorNYI(loc,
"emitLoadOfLValue");
818 const mlir::ArrayAttr elts) {
819 auto elt = mlir::cast<mlir::IntegerAttr>(elts[idx]);
831 if (
getLangOpts().
HLSL && !mlir::isa<cir::VectorType>(vec.getType())) {
832 cgm.errorNYI(loc,
"emitLoadOfExtVectorElementLValue: HLSL");
843 cir::ConstantOp
index =
844 builder.getConstInt(loc, builder.getSInt64Ty(), indexValue);
850 for (
auto i : llvm::seq<unsigned>(0, exprVecTy->getNumElements()))
853 cir::VecShuffleOp resultVec = builder.createVecShuffle(loc, vec, mask);
855 cgm.errorNYI(loc,
"emitLoadOfExtVectorElementLValue: ExtVectorBoolType");
865 "unexpected binary operator opcode");
880 memberPtrTy, &baseInfo);
882 return makeAddrLValue(memberAddr, memberPtrTy->getPointeeType(), baseInfo);
887 mlir::Location loc) {
890 mlir::Type vectorElementTy =
cgm.getTypes().convertType(elementTy);
896 mlir::Value idxValue =
897 builder.getConstInt(loc, mlir::cast<cir::IntType>(
ptrDiffTy), idx);
899 mlir::Value elementValue = builder.getArrayElement(
900 loc, loc, castToPointerElement.
getPointer(), vectorElementTy, idxValue,
906 return Address(elementValue, vectorElementTy, alignment);
915 mlir::Value thisValue) {
923 mlir::Value thisValue) {
924 bool hasExplicitObjectParameter =
false;
925 const auto *methD = dyn_cast_if_present<CXXMethodDecl>(
curCodeDecl);
928 hasExplicitObjectParameter = methD->isExplicitObjectMemberFunction();
929 assert(methD->getParent()->isLambda());
930 assert(methD->getParent() == field->
getParent());
932 if (hasExplicitObjectParameter) {
955 mlir::Type fnTy = funcOp.getFunctionType();
956 mlir::Type ptrTy = cir::PointerType::get(fnTy);
957 mlir::Value addr = cir::GetGlobalOp::create(cgf.
getBuilder(), loc, ptrTy,
958 funcOp.getSymName());
962 ptrTy = cir::PointerType::get(fnTy);
964 addr = cir::CastOp::create(cgf.
getBuilder(), addr.getLoc(), ptrTy,
965 cir::CastKind::bitcast, addr);
1012 case cir::GlobalLinkageKind::ExternalLinkage:
1013 case cir::GlobalLinkageKind::LinkOnceODRLinkage:
1014 case cir::GlobalLinkageKind::WeakODRLinkage:
1015 case cir::GlobalLinkageKind::InternalLinkage:
1016 case cir::GlobalLinkageKind::PrivateLinkage:
1028 "should not emit an unevaluated operand");
1030 if (
const auto *vd = dyn_cast<VarDecl>(nd)) {
1032 if (vd->getStorageClass() ==
SC_Register && vd->hasAttr<AsmLabelAttr>() &&
1033 !vd->isLocalVarDecl()) {
1035 "emitDeclRefLValue: Global Named registers access");
1040 (vd->getType()->isReferenceType() ||
1042 vd->getAnyInitializer(vd);
1044 e->
getLocation(), *vd->evaluateValue(), vd->getType());
1045 assert(val &&
"failed to emit constant expression");
1048 if (!vd->getType()->isReferenceType()) {
1050 addr =
cgm.createUnnamedGlobalFrom(*vd, val,
1053 auto ptrTy = mlir::cast<cir::PointerType>(addr.
getPointer().getType());
1054 if (ptrTy.getPointee() != varTy) {
1073 vd = vd->getCanonicalDecl();
1086 "Should not use decl without marking it used!");
1088 if (nd->
hasAttr<WeakRefAttr>())
1091 if (
const auto *vd = dyn_cast<VarDecl>(nd)) {
1101 if (vd->hasLinkage() || vd->isStaticDataMember())
1109 addr = iter->second;
1111 }
else if (vd->isStaticLocal()) {
1115 cgm.getOrCreateStaticVarDecl(*vd,
cgm.getCIRLinkageVarDefinition(vd));
1116 mlir::Value getGlobVal = builder.createGetGlobal(var);
1117 auto getGlob = getGlobVal.getDefiningOp<cir::GetGlobalOp>();
1118 getGlob.setStaticLocal(var.getStaticLocalGuard().has_value());
1123 llvm_unreachable(
"DeclRefExpr for Decl not entered in localDeclMap?");
1135 assert((vd->isStaticLocal() ||
symbolTable.count(vd)) &&
1136 "non-static locals should be already mapped");
1141 if (
const auto *bd = dyn_cast<BindingDecl>(nd)) {
1149 if (
const auto *fd = dyn_cast<FunctionDecl>(nd)) {
1153 if (
getContext().getTargetInfo().allowDebugInfoForExternalRef())
1160 "emitDeclRefLValue: unhandled MS Guid Decl");
1162 if (
const auto *tpo = dyn_cast<TemplateParamObjectDecl>(nd)) {
1163 CharUnits alignment =
cgm.getNaturalTypeAlignment(tpo->getType());
1164 cir::GetGlobalOp atpo =
1165 builder.createGetGlobal(
cgm.getAddrOfTemplateParamObject(tpo));
1167 "Do an address space conversion if necessary");
1174 llvm_unreachable(
"Unhandled DeclRefExpr");
1184 "evaluateExprAsBool: member pointer type");
1200 if (op == UO_Extension)
1206 assert(!t.
isNull() &&
"CodeGenFunction::EmitUnaryOpLValue: Illegal type");
1226 assert(lv.
isSimple() &&
"real/imag on non-ordinary l-value");
1241 ? builder.createComplexRealPtr(loc, lv.
getAddress())
1242 : builder.createComplexImagPtr(loc, lv.
getAddress());
1252 assert(e->
isPrefix() &&
"Prefix operator in unexpected state!");
1262 llvm_unreachable(
"UnaryOperator extension should be handled above!");
1271 llvm_unreachable(
"UnaryOperator of non-lvalue kind!");
1273 llvm_unreachable(
"Unknown unary operator kind!");
1281 const auto *
castExpr = dyn_cast<CastExpr>(e);
1295 if (
auto constantOp = idx.getDefiningOp<cir::ConstantOp>())
1296 return constantOp.getValueAttr<cir::IntAttr>();
1305 const CharUnits offset = constantIdx.getValue().getZExtValue() * eltSize;
1322 mlir::Location beginLoc,
1323 mlir::Location endLoc, mlir::Value ptr,
1324 mlir::Type eltTy, mlir::Value idx,
1335 mlir::Location beginLoc,
1336 mlir::Location endLoc,
Address addr,
1338 mlir::Location loc,
bool shouldDecay) {
1353 const mlir::Value eltPtr =
1357 return Address(eltPtr, elementType, eltAlign);
1370 "index was neither LHS nor RHS");
1372 auto emitIdxAfterBase = [&](
bool promote) -> mlir::Value {
1381 mlir::Type desiredIdxTy =
1386 if (idx.getType() != desiredIdxTy) {
1387 cir::CastKind
kind = mlir::isa<cir::BoolType>(idx.getType())
1388 ? cir::CastKind::bool_to_int
1389 : cir::CastKind::integral;
1390 idx = builder.createOrFold<cir::CastOp>(idx.getLoc(), desiredIdxTy,
1402 const mlir::Value idx = emitIdxAfterBase(
false);
1415 mlir::Value idx = emitIdxAfterBase(
true);
1440 idx = builder.createIntCast(idx, numElements.getType());
1449 idx = builder.createMul(
cgm.getLoc(e->
getExprLoc()), idx, numElements,
1462 if (
const auto *ase = dyn_cast<ArraySubscriptExpr>(array))
1469 *
this,
cgm.getLoc(array->getBeginLoc()),
cgm.getLoc(array->getEndLoc()),
1484 "The base must be a pointer");
1513 const auto *clangPtrTy =
1515 base =
makeAddrLValue(ptr, clangPtrTy->getPointeeType(), baseInfo);
1516 base.getQuals().removeObjCGCAttr();
1525 "Result must be a vector");
1533 "emitExtVectorElementExpr: ExtVectorBoolType & !HLSL");
1536 builder.createStore(vec.getLoc(), vec, vecMem);
1549 mlir::ArrayAttr elts = builder.getI64ArrayAttr(attrElts);
1559 assert(base.
isExtVectorElt() &&
"Can only subscript lvalue vec elts here!");
1562 for (
unsigned idx : indices)
1564 mlir::ArrayAttr cv = builder.getI64ArrayAttr(elts);
1570 llvm::StringRef name) {
1571 cir::GlobalOp globalOp =
cgm.getGlobalForStringLiteral(e, name);
1572 assert(globalOp.getAlignment() &&
"expected alignment for string literal");
1573 unsigned align = *(globalOp.getAlignment());
1591 case CK_LValueToRValueBitCast:
1592 case CK_ArrayToPointerDecay:
1593 case CK_FunctionToPointerDecay:
1594 case CK_NullToMemberPointer:
1595 case CK_NullToPointer:
1596 case CK_IntegralToPointer:
1597 case CK_PointerToIntegral:
1598 case CK_PointerToBoolean:
1599 case CK_IntegralCast:
1600 case CK_BooleanToSignedIntegral:
1601 case CK_IntegralToBoolean:
1602 case CK_IntegralToFloating:
1603 case CK_FloatingToIntegral:
1604 case CK_FloatingToBoolean:
1605 case CK_FloatingCast:
1606 case CK_FloatingRealToComplex:
1607 case CK_FloatingComplexToReal:
1608 case CK_FloatingComplexToBoolean:
1609 case CK_FloatingComplexCast:
1610 case CK_FloatingComplexToIntegralComplex:
1611 case CK_IntegralRealToComplex:
1612 case CK_IntegralComplexToReal:
1613 case CK_IntegralComplexToBoolean:
1614 case CK_IntegralComplexCast:
1615 case CK_IntegralComplexToFloatingComplex:
1616 case CK_DerivedToBaseMemberPointer:
1617 case CK_BaseToDerivedMemberPointer:
1618 case CK_MemberPointerToBoolean:
1619 case CK_ReinterpretMemberPointer:
1620 case CK_AnyPointerToBlockPointerCast:
1621 case CK_ARCProduceObject:
1622 case CK_ARCConsumeObject:
1623 case CK_ARCReclaimReturnedObject:
1624 case CK_ARCExtendBlockObject:
1625 case CK_CopyAndAutoreleaseBlockObject:
1626 case CK_IntToOCLSampler:
1627 case CK_FloatingToFixedPoint:
1628 case CK_FixedPointToFloating:
1629 case CK_FixedPointCast:
1630 case CK_FixedPointToBoolean:
1631 case CK_FixedPointToIntegral:
1632 case CK_IntegralToFixedPoint:
1634 case CK_HLSLVectorTruncation:
1635 case CK_HLSLMatrixTruncation:
1636 case CK_HLSLArrayRValue:
1637 case CK_HLSLElementwiseCast:
1638 case CK_HLSLAggregateSplatCast:
1639 llvm_unreachable(
"unexpected cast lvalue");
1642 llvm_unreachable(
"dependent cast kind in IR gen!");
1644 case CK_BuiltinFnToFnPtr:
1645 llvm_unreachable(
"builtin functions are handled elsewhere");
1658 case CK_ConstructorConversion:
1659 case CK_UserDefinedConversion:
1660 case CK_CPointerToObjCPointerCast:
1661 case CK_BlockPointerToObjCPointerCast:
1662 case CK_LValueToRValue:
1665 case CK_NonAtomicToAtomic:
1666 case CK_AtomicToNonAtomic:
1667 case CK_ObjCObjectLValueCast:
1668 case CK_VectorSplat: {
1670 std::string(
"emitCastLValue for unhandled cast kind: ") +
1676 case CK_AddressSpaceConversion: {
1681 mlir::ptr::MemorySpaceAttrInterface srcAS;
1687 "emitCastLValue: address space conversion from unknown address "
1697 case CK_LValueBitCast: {
1701 cgm.emitExplicitCastExprType(ce,
this);
1725 auto toTy = dyn_cast<cir::ArrayType>(ty);
1726 if (!fromTy || !toTy ||
1727 fromTy.getElementType() != toTy.getElementType() ||
1728 toTy.getSize() != 0)
1730 "emitCastLValue NoOp not array-shrink case");
1740 case CK_UncheckedDerivedToBase:
1741 case CK_DerivedToBase: {
1759 case CK_BaseToDerived: {
1775 case CK_ZeroToOCLOpaqueType:
1776 llvm_unreachable(
"NULL to OpenCL opaque type lvalue cast is not valid");
1779 llvm_unreachable(
"Invalid cast kind");
1816 if (
auto *field = dyn_cast<FieldDecl>(nd)) {
1828 if (
const auto *fd = dyn_cast<FunctionDecl>(nd))
1831 llvm_unreachable(
"Unhandled member declaration!");
1861 llvm_unreachable(
"bad evaluation kind");
1866 const Expr *inner) {
1877 cir::AllocaOp extDeclAlloca;
1881 extDeclAlloca = extDeclAddrIter->second.getDefiningOp<cir::AllocaOp>();
1883 mlir::OpBuilder::InsertPoint ip;
1884 if (extDeclAlloca) {
1885 ip = {extDeclAlloca->getBlock(), extDeclAlloca->getIterator()};
1903 assert(addr.getAlignment().has_value() &&
1904 "This should always have an alignment");
1910 llvm_unreachable(
"temporary can't have dynamic storage duration");
1912 llvm_unreachable(
"unknown storage duration");
1936 if (
const auto *classDecl =
1940 referenceTemporaryDtor = classDecl->getDestructor();
1942 if (!referenceTemporaryDtor)
1946 "storage duration with destructors");
1962 llvm_unreachable(
"temporary cannot have dynamic storage duration");
1972 "Reference should never be pseudo-strong!");
1980 "emitMaterializeTemporaryExpr: ObjCLifetime");
1988 for (
const Expr *ignored : commaLHSs)
1993 "emitMaterializeTemporaryExpr: OpaqueValueExpr");
2000 if (
auto var =
object.getPointer().getDefiningOp<cir::GlobalOp>()) {
2013 if (!adjustments.empty()) {
2015 "emitMaterializeTemporaryExpr: Adjustments");
2030 assert(e->
isUnique() &&
"LValue for a nonunique OVE hasn't been emitted");
2042 assert(e->
isUnique() &&
"RValue for a nonunique OVE hasn't been emitted");
2056 ".compoundliteral");
2067 "emitCompoundLiteralLValue: non C++ DestructedType");
2082 "Can't have a scalar return unless the return type is a "
2098 assert(e->
getOpcode() == BO_Assign &&
"unexpected binary l-value");
2137 llvm_unreachable(
"bad evaluation kind");
2143 bool ignoreResult) {
2150 if (!ignoreResult && aggSlot.
isIgnored())
2157 llvm_unreachable(
"bad evaluation kind");
2165 if (!pd->isInlineBuiltinDeclaration())
2170CIRGenCallee CIRGenFunction::emitDirectCallee(
const GlobalDecl &gd) {
2173 if (
unsigned builtinID = fd->getBuiltinID()) {
2174 StringRef ident =
cgm.getMangledName(gd);
2175 std::string fdInlineName = (ident +
".inline").str();
2177 bool isPredefinedLibFunction =
2178 cgm.getASTContext().BuiltinInfo.isPredefinedLibFunction(builtinID);
2184 bool hasAttributeNoBuiltin =
false;
2192 mlir::cast_or_null<cir::FuncOp>(
cgm.getGlobalValue(fdInlineName));
2198 mlir::OpBuilder::InsertionGuard guard(builder);
2199 builder.setInsertionPointToStart(
cgm.getModule().getBody());
2201 clone = cir::FuncOp::create(builder, calleeFunc.getLoc(), fdInlineName,
2202 calleeFunc.getFunctionType());
2203 cgm.insertGlobalSymbol(clone);
2204 clone.setLinkageAttr(cir::GlobalLinkageKindAttr::get(
2205 &
cgm.getMLIRContext(), cir::GlobalLinkageKind::InternalLinkage));
2206 clone.setSymVisibility(
"private");
2207 clone.setInlineKind(cir::InlineKind::AlwaysInline);
2218 else if (!isPredefinedLibFunction || !hasAttributeNoBuiltin)
2224 if ((
cgm.getLangOpts().CUDA ||
cgm.getLangOpts().HIP) &&
2225 !
cgm.getLangOpts().CUDAIsDevice && fd->hasAttr<CUDAGlobalAttr>()) {
2226 mlir::Operation *handle =
cgm.getCUDARuntime().getKernelHandle(callee, gd);
2228 mlir::cast<cir::FuncOp>(*
cgm.getCUDARuntime().getKernelStub(handle));
2238 cgm.errorNYI(
"unsupported type for undef rvalue");
2249 "Callee must have function pointer type!");
2270 cgm.getTypes().arrangeFreeFunctionCall(args, fnType);
2292 calleeTy = cir::FuncType::get(calleeTy.getInputs(),
2293 calleeTy.getReturnType(),
false);
2294 auto calleePtrTy = cir::PointerType::get(calleeTy);
2298 if (
auto funcOp = mlir::dyn_cast<cir::FuncOp>(fn)) {
2299 addr = cir::GetGlobalOp::create(
2301 cir::PointerType::get(funcOp.getFunctionType()), funcOp.getSymName());
2303 addr = fn->getResult(0);
2306 fn = builder.createBitcast(addr, calleePtrTy).getDefiningOp();
2314 cir::CIRCallOpInterface callOp;
2327 if (
const auto *implicitCast = dyn_cast<ImplicitCastExpr>(e)) {
2328 if (implicitCast->getCastKind() == CK_FunctionToPointerDecay ||
2329 implicitCast->getCastKind() == CK_BuiltinFnToFnPtr) {
2330 return emitCallee(implicitCast->getSubExpr());
2335 assert(implicitCast->getCastKind() == CK_LValueToRValue &&
2336 "unexpected implicit cast on function pointers");
2337 }
else if (
const auto *declRef = dyn_cast<DeclRefExpr>(e)) {
2340 return emitDirectCallee(funcDecl);
2341 }
else if (
auto me = dyn_cast<MemberExpr>(e)) {
2342 if (
const auto *fd = dyn_cast<FunctionDecl>(me->getMemberDecl())) {
2344 return emitDirectCallee(fd);
2347 }
else if (
auto *pde = dyn_cast<CXXPseudoDestructorExpr>(e)) {
2352 mlir::Value calleePtr;
2364 if (
const auto *vd =
2369 CIRGenCallee callee(calleeInfo, calleePtr.getDefiningOp());
2377 if (
const auto *ce = dyn_cast<CXXMemberCallExpr>(e))
2383 if (
const auto *operatorCall = dyn_cast<CXXOperatorCallExpr>(e)) {
2387 dyn_cast_or_null<CXXMethodDecl>(operatorCall->getCalleeDecl()))
2421 "Array to pointer decay must have array source type!");
2429 auto lvalueAddrTy = mlir::cast<cir::PointerType>(addr.
getPointer().getType());
2434 [[maybe_unused]]
auto pointeeTy =
2435 mlir::cast<cir::ArrayType>(lvalueAddrTy.getPointee());
2438 assert(mlir::isa<cir::ArrayType>(arrayTy) &&
"expected array");
2439 assert(pointeeTy == arrayTy);
2449 mlir::Value ptr = builder.maybeBuildArrayDecay(
2467 llvm_unreachable(
"bad evaluation kind");
2474 const Stmt *elseS) {
2476 std::optional<mlir::Location> elseLoc;
2480 mlir::LogicalResult resThen = mlir::success(), resElse = mlir::success();
2483 [&](mlir::OpBuilder &, mlir::Location) {
2484 LexicalScope lexScope{*
this, thenLoc, builder.getInsertionBlock()};
2489 [&](mlir::OpBuilder &, mlir::Location) {
2490 assert(elseLoc &&
"Invalid location for elseS.");
2491 LexicalScope lexScope{*
this, *elseLoc, builder.getInsertionBlock()};
2496 return mlir::LogicalResult::success(resThen.succeeded() &&
2497 resElse.succeeded());
2505 std::optional<mlir::Location> elseLoc) {
2511 ifLocs.push_back(*elseLoc);
2512 mlir::Location loc = mlir::FusedLoc::get(&
getMLIRContext(), ifLocs);
2516 cir::IfOp ifOp = cir::IfOp::create(builder, loc, condV, elseLoc.has_value(),
2520 assert((elseLoc.has_value() || ifOp.getElseRegion().empty()) &&
2521 "else region created with no else location");
2522 if (elseLoc.has_value())
2541 Expr *trueExpr = condOp->getTrueExpr();
2542 Expr *falseExpr = condOp->getFalseExpr();
2545 mlir::Value ternaryOpRes =
2546 cir::TernaryOp::create(
2547 builder, loc, condV,
2548 [
this, trueExpr](mlir::OpBuilder &
b, mlir::Location loc) {
2550 cir::YieldOp::create(
b, loc, lhs);
2553 [
this, falseExpr](mlir::OpBuilder &
b, mlir::Location loc) {
2555 cir::YieldOp::create(
b, loc, rhs);
2564 cgm.errorNYI(
"NYI");
2578 mlir::Location loc,
CharUnits alignment,
2579 bool insertIntoFnEntryBlock,
2580 mlir::Value arraySize) {
2581 mlir::Block *entryBlock = insertIntoFnEntryBlock
2590 llvm::dyn_cast_if_present<cir::TryOp>(entryBlock->getParentOp())) {
2591 if (auto scopeOp = llvm::dyn_cast<cir::ScopeOp>(tryOp->getParentOp()))
2592 entryBlock = &scopeOp.getScopeRegion().front();
2600 mlir::Location loc,
CharUnits alignment,
2601 mlir::OpBuilder::InsertPoint ip,
2602 mlir::Value arraySize) {
2606 cir::PointerType localVarPtrTy =
2608 mlir::IntegerAttr alignIntAttr =
cgm.
getSize(alignment);
2612 mlir::OpBuilder::InsertionGuard guard(builder);
2613 builder.restoreInsertionPoint(ip);
2615 ty, name, alignIntAttr, arraySize);
2633 if (md->isStatic()) {
2638 bool hasQualifier = me->hasQualifier();
2640 bool isArrow = me->isArrow();
2641 const Expr *base = me->getBase();
2644 ce, md,
returnValue, hasQualifier, qualifier, isArrow, base);
2661 cgm.errorNYI(loc,
"load of volatile reference");
2670 CharUnits align =
cgm.getNaturalTypeAlignment(pointeeType, pointeeBaseInfo);
2687 cir::TrapOp::create(builder, loc);
2689 builder.createBlock(builder.getBlock()->getParent());
2693 bool createNewBlock) {
2695 cir::UnreachableOp::create(builder,
getLoc(loc));
2697 builder.createBlock(builder.getBlock()->getParent());
2704 return builder.createDummyValue(loc, t, alignment);
2712 const Twine &name,
Address *alloca,
2713 mlir::OpBuilder::InsertPoint ip) {
2720 mlir::Location loc,
const Twine &name,
2722 mlir::OpBuilder::InsertPoint ip) {
2724 nullptr, alloca, ip);
2727 cgm.errorNYI(loc,
"temporary matrix value");
2735 mlir::Type ty,
CharUnits align, mlir::Location loc,
const Twine &name,
2736 mlir::Value arraySize, mlir::OpBuilder::InsertPoint ip) {
2737 cir::AllocaOp alloca = ip.isSet()
2740 alloca.setAlignmentAttr(
cgm.getSize(align));
2741 return Address(alloca, ty, align);
2749 mlir::Location loc,
const Twine &name,
2750 mlir::Value arraySize,
2752 mlir::OpBuilder::InsertPoint ip) {
2756 *allocaAddr = alloca;
2763 cir::PointerType dstTy;
2779 mlir::Value arraySize,
2780 bool insertIntoFnEntryBlock) {
2782 insertIntoFnEntryBlock, arraySize)
2790 mlir::OpBuilder::InsertPoint ip,
2791 mlir::Value arraySize) {
2792 assert(ip.isSet() &&
"Insertion point is not set");
2793 return mlir::cast<cir::AllocaOp>(
2804 const Twine &name) {
2846 mlir::TypedAttr cstToEmit = mlir::dyn_cast_if_present<mlir::TypedAttr>(
c);
2847 assert(cstToEmit &&
"expected a typed attribute");
2863 assert(constant &&
"not a constant");
2873 assert(sl !=
nullptr &&
"No StringLiteral name in PredefinedExpr");
2875 StringRef fnName = fn.getName();
2876 fnName.consume_front(
"\01");
2877 std::array<StringRef, 2> nameItems = {
2879 std::string gvName = llvm::join(nameItems,
".");
2894std::optional<LValue> handleConditionalOperatorLValueSimpleCase(
2897 llvm::APSInt condExprVal;
2899 return std::nullopt;
2902 if (!condExprVal.getBoolValue())
2903 std::swap(live, dead);
2906 return std::nullopt;
2913 if (
auto *throwExpr = dyn_cast<CXXThrowExpr>(live->
IgnoreParens())) {
2917 mlir::Type elemTy = cgf.
convertType(dead->getType());
2918 mlir::Value undefPtr =
2920 cgf.
getLoc(throwExpr->getSourceRange()));
2930static std::optional<LValue> emitLValueOrThrowExpression(
CIRGenFunction &cgf,
2931 const Expr *operand) {
2932 if (
auto *throwExpr = dyn_cast<CXXThrowExpr>(operand->
IgnoreParens())) {
2934 return std::nullopt;
2943template <
typename FuncTy>
2946 const FuncTy &branchGenFunc) {
2954 mlir::Type yieldTy{};
2956 auto emitBranch = [&](mlir::OpBuilder &
b, mlir::Location loc,
2957 const Expr *
expr, std::optional<LValue> &resultLV) {
2963 resultLV = branchGenFunc(*
this,
expr);
2964 mlir::Value resultPtr = resultLV ? resultLV->getPointer() : mlir::Value();
2968 yieldTy = resultPtr.getType();
2969 cir::YieldOp::create(
b, loc, resultPtr);
2975 if (builder.getInsertionBlock()->mightHaveTerminator()) {
2976 mlir::Operation *terminator =
2977 builder.getInsertionBlock()->getTerminator();
2978 if (isa_and_nonnull<cir::UnreachableOp>(terminator))
2979 insertPoints.push_back(
b.saveInsertionPoint());
2984 info.
result = cir::TernaryOp::create(
2985 builder, loc, condV,
2987 [&](mlir::OpBuilder &
b, mlir::Location loc) {
2991 [&](mlir::OpBuilder &
b, mlir::Location loc) {
3001 for (mlir::OpBuilder::InsertPoint &toInsert : insertPoints) {
3002 mlir::OpBuilder::InsertionGuard guard(builder);
3003 builder.restoreInsertionPoint(toInsert);
3007 cir::YieldOp::create(builder, loc);
3009 mlir::Value op0 = builder.getNullValue(yieldTy, loc);
3010 cir::YieldOp::create(builder, loc, op0);
3019 if (!
expr->isGLValue()) {
3022 "Unexpected conditional operator!");
3027 if (std::optional<LValue> res =
3028 handleConditionalOperatorLValueSimpleCase(*
this,
expr))
3033 return emitLValueOrThrowExpression(cgf, e);
3036 if ((info.
lhs && !info.
lhs->isSimple()) ||
3037 (info.
rhs && !info.
rhs->isSimple())) {
3038 cgm.errorNYI(
expr->getSourceRange(),
3039 "unsupported conditional operator with non-simple lvalue");
3043 if (info.
lhs && info.
rhs) {
3049 std::max(info.
lhs->getBaseInfo().getAlignmentSource(),
3050 info.
rhs->getBaseInfo().getAlignmentSource());
3055 assert((info.
lhs || info.
rhs) &&
3056 "both operands of glvalue conditional are throw-expressions?");
3057 return info.
lhs ? *info.
lhs : *info.
rhs;
3064 if (!
cgm.getLangOpts().MSVolatile)
3067 cgm.errorNYI(
"LValueSuitableForInlineAtomic LangOpts MSVolatile");
Provides definitions for the various language-specific address spaces.
llvm::function_ref< void(mlir::OpBuilder &, mlir::Location)> BuilderCallbackRef
static Address createReferenceTemporary(CIRGenFunction &cgf, const MaterializeTemporaryExpr *m, const Expr *inner)
static bool isAAPCS(const TargetInfo &targetInfo)
Helper method to check if the underlying ABI is AAPCS.
static LValue emitFunctionDeclLValue(CIRGenFunction &cgf, const Expr *e, GlobalDecl gd)
static CharUnits getArrayElementAlign(CharUnits arrayAlign, mlir::Value idx, CharUnits eltSize)
static void pushTemporaryCleanup(CIRGenFunction &cgf, const MaterializeTemporaryExpr *m, const Expr *e, Address referenceTemporary)
static cir::IntAttr getConstantIndexOrNull(mlir::Value idx)
static const Expr * getSimpleArrayDecayOperand(const Expr *e)
If the specified expr is a simple decay from an array to pointer, return the array subexpression.
static QualType getFixedSizeElementType(const ASTContext &astContext, const VariableArrayType *vla)
static bool canEmitSpuriousReferenceToVariable(CIRGenFunction &cgf, const DeclRefExpr *e, const VarDecl *vd)
Determine whether we can emit a reference to vd from the current context, despite not necessarily hav...
static DeclRefExpr * tryToConvertMemberExprToDeclRefExpr(CIRGenFunction &cgf, const MemberExpr *me)
static cir::FuncOp emitFunctionDeclPointer(CIRGenModule &cgm, GlobalDecl gd)
static LValue emitGlobalVarDeclLValue(CIRGenFunction &cgf, const Expr *e, const VarDecl *vd)
static mlir::Value emitArraySubscriptPtr(CIRGenFunction &cgf, mlir::Location beginLoc, mlir::Location endLoc, mlir::Value ptr, mlir::Type eltTy, mlir::Value idx, bool shouldDecay)
static LValue emitCapturedFieldLValue(CIRGenFunction &cgf, const FieldDecl *fd, mlir::Value thisValue)
static bool onlyHasInlineBuiltinDeclaration(const FunctionDecl *fd)
static Address emitAddrOfZeroSizeField(CIRGenFunction &cgf, Address base, const FieldDecl *field)
Get the address of a zero-sized field within a record.
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 c
cir::ConstantOp getConstant(mlir::Location loc, mlir::TypedAttr attr)
cir::PtrStrideOp createPtrStride(mlir::Location loc, mlir::Value base, mlir::Value stride)
static OpBuilder::InsertPoint getBestAllocaInsertPoint(mlir::Block *block)
cir::GetMemberOp createGetMember(mlir::Location loc, mlir::Type resultTy, mlir::Value base, llvm::StringRef name, unsigned index)
cir::PointerType getPointerTo(mlir::Type ty)
cir::ConstantOp getNullPtr(mlir::Type ty, mlir::Location loc)
mlir::Value createPtrBitcast(mlir::Value src, mlir::Type newPointeeTy)
cir::GetGlobalOp createGetGlobal(mlir::Location loc, cir::GlobalOp global, bool threadLocal=false)
mlir::Value createBitcast(mlir::Value src, mlir::Type newTy)
mlir::Value createAlloca(mlir::Location loc, cir::PointerType addrType, mlir::Type type, llvm::StringRef name, mlir::IntegerAttr alignment, mlir::Value dynAllocSize)
llvm::TypeSize getTypeSizeInBits(mlir::Type ty) const
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
uint64_t getFieldOffset(const ValueDecl *FD) const
Get the offset of a FieldDecl or IndirectFieldDecl, in bits.
static CanQualType getCanonicalType(QualType T)
Return the canonical (structural) type corresponding to the specified potentially non-canonical type ...
QualType getPointerType(QualType T) const
Return the uniqued reference to the type for a pointer to the specified type.
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
const VariableArrayType * getAsVariableArrayType(QualType T) const
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
CanQualType getCanonicalTagType(const TagDecl *TD) const
AbstractConditionalOperator - An abstract base class for ConditionalOperator and BinaryConditionalOpe...
Expr * getCond() const
getCond - Return the expression representing the condition for the ?
Expr * getTrueExpr() const
getTrueExpr - Return the subexpression representing the value of the expression if the condition eval...
Expr * getFalseExpr() const
getFalseExpr - Return the subexpression representing the value of the expression if the condition eva...
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
SourceLocation getExprLoc() const LLVM_READONLY
Expr * getLHS()
An array access can be written A[4] or 4[A] (both are equivalent).
SourceLocation getBeginLoc() const LLVM_READONLY
SourceLocation getEndLoc() const
QualType getElementType() const
A builtin binary operation expression such as "x + y" or "x <= y".
mlir::Value getPointer() const
mlir::Type getElementType() const
Address withElementType(CIRGenBuilderTy &builder, mlir::Type ElemTy) const
Return address with different element type, a bitcast pointer, and the same alignment.
clang::CharUnits getAlignment() const
mlir::Type getType() const
mlir::Operation * getDefiningOp() const
Get the operation which defines this address.
IsDestructed_t
This is set to true if the slot might be aliased and it's not undefined behavior to access it through...
static AggValueSlot forAddr(Address addr, clang::Qualifiers quals, IsDestructed_t isDestructed, IsAliased_t isAliased, Overlap_t mayOverlap, IsZeroed_t isZeroed=IsNotZeroed)
static AggValueSlot ignored()
Returns an aggregate value slot indicating that the aggregate value is being ignored.
cir::IntType getUInt8Ty()
cir::ConstantOp getUInt64(uint64_t c, mlir::Location loc)
Address createElementBitCast(mlir::Location loc, Address addr, mlir::Type destType)
Cast the element type of the given address to a different type, preserving information like the align...
mlir::Value getArrayElement(mlir::Location arrayLocBegin, mlir::Location arrayLocEnd, mlir::Value arrayPtr, mlir::Type eltTy, mlir::Value idx, bool shouldDecay)
Create a cir.ptr_stride operation to get access to an array element.
Abstract information about a function or function prototype.
bool isPseudoDestructor() const
void setFunctionPointer(mlir::Operation *functionPtr)
const clang::FunctionDecl * getBuiltinDecl() const
const CXXPseudoDestructorExpr * getPseudoDestructorExpr() const
static CIRGenCallee forDirect(mlir::Operation *funcPtr, const CIRGenCalleeInfo &abstractInfo=CIRGenCalleeInfo())
unsigned getBuiltinID() const
static CIRGenCallee forBuiltin(unsigned builtinID, const clang::FunctionDecl *builtinDecl)
mlir::Operation * getFunctionPointer() const
static CIRGenCallee forPseudoDestructor(const clang::CXXPseudoDestructorExpr *expr)
An object to manage conditionally-evaluated expressions.
static ConstantEmission forValue(mlir::TypedAttr c)
mlir::TypedAttr getValue() const
static bool shouldBindAsLValue(const Expr *expr)
An RAII object to set (and then clear) a mapping for an OpaqueValueExpr.
static bool shouldBindAsLValue(const Expr *expr)
mlir::Value emitComplexToScalarConversion(mlir::Value src, QualType srcTy, QualType dstTy, SourceLocation loc)
Emit a conversion from the specified complex type to the specified destination type,...
void emitCallArgs(CallArgList &args, PrototypeWrapper prototype, llvm::iterator_range< clang::CallExpr::const_arg_iterator > argRange, AbstractCallee callee=AbstractCallee(), unsigned paramsToSkip=0)
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.
RValue convertTempToRValue(Address addr, clang::QualType type, clang::SourceLocation loc)
Given the address of a temporary variable, produce an r-value of its type.
Address emitCXXMemberDataPointerAddress(const Expr *e, Address base, mlir::Value memberPtr, const MemberPointerType *memberPtrType, LValueBaseInfo *baseInfo)
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
cir::AllocaOp createTempAlloca(mlir::Type ty, mlir::Location loc, const Twine &name="tmp", mlir::Value arraySize=nullptr, bool insertIntoFnEntryBlock=false)
This creates an alloca and inserts it into the entry block if ArraySize is nullptr,...
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...
mlir::Block * getCurFunctionEntryBlock()
RValue emitCXXMemberCallExpr(const clang::CXXMemberCallExpr *e, ReturnValueSlot returnValue)
RValue emitCXXMemberPointerCallExpr(const CXXMemberCallExpr *ce, ReturnValueSlot returnValue)
bool isInConditionalBranch() const
LValue emitLValueForBitField(LValue base, const FieldDecl *field)
mlir::LogicalResult emitIfOnBoolExpr(const clang::Expr *cond, const clang::Stmt *thenS, const clang::Stmt *elseS)
Emit an if on a boolean condition to the specified blocks.
VlaSizePair getVLASize(const VariableArrayType *type)
Returns an MLIR::Value+QualType pair that corresponds to the size, in non-variably-sized elements,...
mlir::Value emitComplexExpr(const Expr *e)
Emit the computation of the specified expression of complex type, returning the result.
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...
RValue emitCallExpr(const clang::CallExpr *e, ReturnValueSlot returnValue=ReturnValueSlot())
LValue emitMemberExpr(const MemberExpr *e)
LValue emitConditionalOperatorLValue(const AbstractConditionalOperator *expr)
LValue emitLValue(const clang::Expr *e)
Emit code to compute a designator that specifies the location of the expression.
Address makeNaturalAddressForPointer(mlir::Value ptr, QualType t, CharUnits alignment, bool forPointeeType=false, LValueBaseInfo *baseInfo=nullptr)
Construct an address with the natural alignment of T.
LValue emitLValueForLambdaField(const FieldDecl *field)
mlir::Value evaluateExprAsBool(const clang::Expr *e)
Perform the usual unary conversions on the specified expression and compare the result against zero,...
std::string getCounterRefTmpAsString()
LValue makeNaturalAlignAddrLValue(mlir::Value val, QualType ty)
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,...
mlir::Location getLoc(clang::SourceLocation srcLoc)
Helpers to convert Clang's SourceLocation to a MLIR Location.
mlir::Value emitOpOnBoolExpr(mlir::Location loc, const clang::Expr *cond)
TODO(cir): see EmitBranchOnBoolExpr for extra ideas).
void emitStoreThroughExtVectorComponentLValue(RValue src, LValue dst)
Address getAddressOfBaseClass(Address value, const CXXRecordDecl *derived, llvm::iterator_range< CastExpr::path_const_iterator > path, bool nullCheckValue, SourceLocation loc)
LValue emitLoadOfReferenceLValue(Address refAddr, mlir::Location loc, QualType refTy, AlignmentSource source)
void emitAnyExprToMem(const Expr *e, Address location, Qualifiers quals, bool isInitializer)
Emits the code necessary to evaluate an arbitrary expression into the given memory location.
LValue getOrCreateOpaqueLValueMapping(const OpaqueValueExpr *e)
Given an opaque value expression, return its LValue mapping if it exists, otherwise create one.
mlir::Value performAddrSpaceCast(mlir::Value v, mlir::Type destTy) const
RValue emitReferenceBindingToExpr(const Expr *e)
Emits a reference binding to the passed in expression.
LValue emitArraySubscriptExpr(const clang::ArraySubscriptExpr *e)
RValue emitCUDAKernelCallExpr(const CUDAKernelCallExpr *expr, ReturnValueSlot returnValue)
mlir::Operation * curFn
The current function or global initializer that is generated code for.
Address emitExtVectorElementLValue(LValue lv, mlir::Location loc)
Generates lvalue for partial ext_vector access.
mlir::Value emitScalarConversion(mlir::Value src, clang::QualType srcType, clang::QualType dstType, clang::SourceLocation loc)
Emit a conversion from the specified type to the specified destination type, both of which are CIR sc...
Address getAddressOfDerivedClass(mlir::Location loc, Address baseAddr, const CXXRecordDecl *derived, llvm::iterator_range< CastExpr::path_const_iterator > path, bool nullCheckValue)
RValue emitAtomicLoad(LValue lvalue, SourceLocation loc, AggValueSlot slot=AggValueSlot::ignored())
AggValueSlot createAggTemp(QualType ty, mlir::Location loc, const Twine &name="tmp", Address *alloca=nullptr)
Create a temporary memory object for the given aggregate type.
RValue emitLoadOfExtVectorElementLValue(LValue lv)
mlir::Value emitCXXTypeidExpr(const CXXTypeidExpr *e)
mlir::Type convertTypeForMem(QualType t)
mlir::Value emitAlloca(llvm::StringRef name, mlir::Type ty, mlir::Location loc, clang::CharUnits alignment, bool insertIntoFnEntryBlock, mlir::Value arraySize=nullptr)
mlir::Value emitComplexPrePostIncDec(const UnaryOperator *e, LValue lv)
void emitUnreachable(clang::SourceLocation loc, bool createNewBlock)
Emit a reached-unreachable diagnostic if loc is valid and runtime checking is enabled.
mlir::Value createDummyValue(mlir::Location loc, clang::QualType qt)
mlir::Value emitLoadOfComplex(LValue src, SourceLocation loc)
Load a complex number from the specified l-value.
LValue emitAggExprToLValue(const Expr *e)
void emitStoreOfScalar(mlir::Value value, Address addr, bool isVolatile, clang::QualType ty, LValueBaseInfo baseInfo, bool isInit=false, bool isNontemporal=false)
void pushDestroy(QualType::DestructionKind dtorKind, Address addr, QualType type)
Push the standard destructor for the given type as at least a normal cleanup.
static Destroyer destroyCXXObject
RValue getUndefRValue(clang::QualType ty)
Get an appropriate 'undef' rvalue for the given type.
Address returnValue
The temporary alloca to hold the return value.
static bool hasAggregateEvaluationKind(clang::QualType type)
LValue emitPointerToDataMemberBinaryExpr(const BinaryOperator *e)
std::string getCounterAggTmpAsString()
LValue emitUnaryOpLValue(const clang::UnaryOperator *e)
RValue emitLoadOfBitfieldLValue(LValue lv, SourceLocation loc)
RValue emitCall(const CIRGenFunctionInfo &funcInfo, const CIRGenCallee &callee, ReturnValueSlot returnValue, const CallArgList &args, cir::CIRCallOpInterface *callOp, mlir::Location loc)
LValue emitComplexAssignmentLValue(const BinaryOperator *e)
const clang::Decl * curCodeDecl
This is the inner-most code context, which includes blocks.
LValue emitLValueForFieldInitialization(LValue base, const clang::FieldDecl *field, llvm::StringRef fieldName)
Like emitLValueForField, excpet that if the Field is a reference, this will return the address of the...
LValue emitCallExprLValue(const clang::CallExpr *e)
LValue emitStringLiteralLValue(const StringLiteral *e, llvm::StringRef name=".str")
mlir::Value emitToMemory(mlir::Value value, clang::QualType ty)
Given a value and its clang type, returns the value casted to its memory representation.
LValue emitLValueForField(LValue base, const clang::FieldDecl *field)
mlir::Value emitScalarExpr(const clang::Expr *e, bool ignoreResultAssign=false)
Emit the computation of the specified expression of scalar type.
Address emitLoadOfReference(LValue refLVal, mlir::Location loc, LValueBaseInfo *pointeeBaseInfo)
bool shouldNullCheckClassCastValue(const CastExpr *ce)
CIRGenBuilderTy & getBuilder()
LValue emitBinaryOperatorLValue(const BinaryOperator *e)
Address getAddrOfBitFieldStorage(LValue base, const clang::FieldDecl *field, mlir::Type fieldType, unsigned index)
CIRGenModule & getCIRGenModule()
mlir::MLIRContext & getMLIRContext()
LValue emitCastLValue(const CastExpr *e)
Casts are never lvalues unless that cast is to a reference type.
LValue emitCXXTypeidLValue(const CXXTypeidExpr *e)
mlir::Value emitLoadOfScalar(LValue lvalue, SourceLocation loc)
EmitLoadOfScalar - Load a scalar value from an address, taking care to appropriately convert from the...
mlir::Value emitScalarPrePostIncDec(const UnaryOperator *e, LValue lv)
bool containsLabel(const clang::Stmt *s, bool ignoreCaseStmts=false)
Return true if the statement contains a label in it.
DeclMapTy localDeclMap
This keeps track of the CIR allocas or globals for local C declarations.
LValue emitDeclRefLValue(const clang::DeclRefExpr *e)
void emitComplexExprIntoLValue(const Expr *e, LValue dest, bool isInit)
llvm::DenseMap< const clang::ValueDecl *, clang::FieldDecl * > lambdaCaptureFields
ConstantEmission tryEmitAsConstant(const DeclRefExpr *refExpr)
Try to emit a reference to the given value without producing it as an l-value.
void emitCXXThrowExpr(const CXXThrowExpr *e)
LValue makeAddrLValue(Address addr, QualType ty, AlignmentSource source=AlignmentSource::Type)
int64_t getAccessedFieldNo(unsigned idx, mlir::ArrayAttr elts)
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.
llvm::DenseMap< const OpaqueValueExpr *, RValue > opaqueRValues
Address emitArrayToPointerDecay(const Expr *e, LValueBaseInfo *baseInfo=nullptr)
LexicalScope * curLexScope
void emitAtomicStore(RValue rvalue, LValue dest, bool isInit)
mlir::Value emitStoreThroughBitfieldLValue(RValue src, LValue dstresult)
llvm::DenseMap< const OpaqueValueExpr *, LValue > opaqueLValues
Keeps track of the current set of opaque value expressions.
CIRGenFunction(CIRGenModule &cgm, CIRGenBuilderTy &builder, bool suppressNewContext=false)
std::optional< mlir::Location > currSrcLoc
Use to track source locations across nested visitor traversals.
void terminateStructuredRegionBody(mlir::Region &r, mlir::Location loc)
LValue emitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *e)
LValue emitExtVectorElementExpr(const ExtVectorElementExpr *e)
clang::ASTContext & getContext() const
RValue emitCXXMemberOrOperatorMemberCallExpr(const clang::CallExpr *ce, const clang::CXXMethodDecl *md, ReturnValueSlot returnValue, bool hasQualifier, clang::NestedNameSpecifier qualifier, bool isArrow, const clang::Expr *base)
mlir::Value emitScalarConstant(const ConstantEmission &constant, Expr *e)
RValue emitBuiltinExpr(const clang::GlobalDecl &gd, unsigned builtinID, const clang::CallExpr *e, ReturnValueSlot returnValue)
RValue emitCXXOperatorMemberCallExpr(const CXXOperatorCallExpr *e, const CXXMethodDecl *md, ReturnValueSlot returnValue)
void emitStoreThroughLValue(RValue src, LValue dst, bool isInit=false)
Store the specified rvalue into the specified lvalue, where both are guaranteed to the have the same ...
mlir::Value cxxabiThisValue
bool isLValueSuitableForInlineAtomic(LValue lv)
An LValue is a candidate for having its loads and stores be made atomic if we are operating under /vo...
mlir::LogicalResult emitStmt(const clang::Stmt *s, bool useCurrentScope, llvm::ArrayRef< const Attr * > attrs={})
void pushLifetimeExtendedDestroy(CleanupKind kind, Address addr, QualType type, Destroyer *destroyer, bool useEHCleanupForArray)
RValue getOrCreateOpaqueRValueMapping(const OpaqueValueExpr *e)
Given an opaque value expression, return its RValue mapping if it exists, otherwise create one.
Address createTempAllocaWithoutCast(mlir::Type ty, CharUnits align, mlir::Location loc, const Twine &name="tmp", mlir::Value arraySize=nullptr, mlir::OpBuilder::InsertPoint ip={})
This creates a alloca and inserts it into the entry block of the current region.
mlir::Value emitFromMemory(mlir::Value value, clang::QualType ty)
EmitFromMemory - Change a scalar value from its memory representation to its value representation.
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...
mlir::Value emitDynamicCast(Address thisAddr, const CXXDynamicCastExpr *dce)
void emitAggExpr(const clang::Expr *e, AggValueSlot slot)
ConditionalInfo emitConditionalBlocks(const AbstractConditionalOperator *e, const FuncTy &branchGenFunc)
Address createDefaultAlignTempAlloca(mlir::Type ty, mlir::Location loc, const Twine &name)
CreateDefaultAlignTempAlloca - This creates an alloca with the default alignment of the corresponding...
RValue emitCXXPseudoDestructorExpr(const CXXPseudoDestructorExpr *expr)
LValue emitCompoundLiteralLValue(const CompoundLiteralExpr *e)
CIRGenCallee emitCallee(const clang::Expr *e)
Address emitAddrOfFieldStorage(Address base, const FieldDecl *field, llvm::StringRef fieldName, unsigned fieldIndex)
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.
cir::GlobalLinkageKind getCIRLinkageVarDefinition(const VarDecl *vd)
mlir::IntegerAttr getSize(CharUnits size)
CIRGenBuilderTy & getBuilder()
cir::FuncOp getAddrOfFunction(clang::GlobalDecl gd, mlir::Type funcType=nullptr, bool forVTable=false, bool dontDefer=false, ForDefinition_t isForDefinition=NotForDefinition)
Return the address of the given function.
mlir::Operation * getAddrOfGlobalTemporary(const MaterializeTemporaryExpr *mte, const Expr *init)
Returns a pointer to a global variable representing a temporary with static or thread storage duratio...
mlir::Value getAddrOfGlobalVar(const VarDecl *d, mlir::Type ty={}, ForDefinition_t isForDefinition=NotForDefinition)
Return the mlir::Value for the address of the given global variable.
This class handles record and union layout info while lowering AST types to CIR types.
cir::RecordType getCIRType() const
Return the "complete object" LLVM type associated with this record.
const CIRGenBitFieldInfo & getBitFieldInfo(const clang::FieldDecl *fd) const
Return the BitFieldInfo that corresponds to the field FD.
unsigned getCIRFieldNo(const clang::FieldDecl *fd) const
Return cir::RecordType element number that corresponds to the field FD.
cir::FuncType getFunctionType(const CIRGenFunctionInfo &info)
Get the CIR function type for.
mlir::Type convertTypeForMem(clang::QualType, bool forBitField=false)
Convert type T into an mlir::Type.
mlir::Attribute emitAbstract(const Expr *e, QualType destType)
Emit the result of the given expression as an abstract constant, asserting that it succeeded.
AlignmentSource getAlignmentSource() const
void mergeForCast(const LValueBaseInfo &info)
bool isExtVectorElt() const
mlir::Value getVectorPointer() const
const clang::Qualifiers & getQuals() const
mlir::Value getExtVectorPointer() const
static LValue makeExtVectorElt(Address vecAddress, mlir::ArrayAttr elts, clang::QualType type, LValueBaseInfo baseInfo)
mlir::Value getVectorIdx() const
Address getAddress() const
static LValue makeAddr(Address address, clang::QualType t, LValueBaseInfo baseInfo)
mlir::ArrayAttr getExtVectorElts() const
static LValue makeVectorElt(Address vecAddress, mlir::Value index, clang::QualType t, LValueBaseInfo baseInfo)
RValue asAggregateRValue() const
unsigned getVRQualifiers() const
clang::QualType getType() const
static LValue makeBitfield(Address addr, const CIRGenBitFieldInfo &info, clang::QualType type, LValueBaseInfo baseInfo)
Create a new object to represent a bit-field access.
mlir::Value getPointer() const
bool isVolatileQualified() const
Address getVectorAddress() const
clang::CharUnits getAlignment() const
LValueBaseInfo getBaseInfo() const
const CIRGenBitFieldInfo & getBitFieldInfo() const
Address getBitFieldAddress() const
Address getExtVectorAddress() const
This trivial value class is used to represent the result of an expression that is evaluated.
Address getAggregateAddress() const
Return the value of the address of the aggregate.
static RValue get(mlir::Value v)
static RValue getComplex(mlir::Value v)
mlir::Value getValue() const
Return the value of this scalar value.
Contains the address where the return value of a function can be stored, and whether the address is v...
Represents a C++ destructor within a class.
Represents a call to a member function that may be written either with member call syntax (e....
Represents a static or instance method of a struct/union/class.
Represents a C++ struct/union/class.
bool hasTrivialDestructor() const
Determine whether this class has a trivial destructor (C++ [class.dtor]p3)
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
FunctionDecl * getDirectCallee()
If the callee is a FunctionDecl, return it. Otherwise return null.
QualType getCallReturnType(const ASTContext &Ctx) const
getCallReturnType - Get the return type of the call expr.
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
CastKind getCastKind() const
llvm::iterator_range< path_iterator > path()
Path through the class hierarchy taken by casts between base and derived classes (see implementation ...
bool changesVolatileQualification() const
Return.
static const char * getCastKindName(CastKind CK)
CharUnits - This is an opaque type for sizes expressed in character units.
CharUnits alignmentAtOffset(CharUnits offset) const
Given that this is a non-zero alignment value, what is the alignment at the given offset?
bool isZero() const
isZero - Test whether the quantity equals zero.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
CharUnits alignmentOfArrayElement(CharUnits elementSize) const
Given that this is the alignment of the first element of an array, return the minimum alignment of an...
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
Complex values, per C99 6.2.5p11.
CompoundLiteralExpr - [C99 6.5.2.5].
const Expr * getInitializer() const
ConditionalOperator - The ?
A reference to a declared variable, function, enum, etc.
bool refersToEnclosingVariableOrCapture() const
Does this DeclRefExpr refer to an enclosing local or a captured variable?
static DeclRefExpr * Create(const ASTContext &Context, NestedNameSpecifierLoc QualifierLoc, SourceLocation TemplateKWLoc, ValueDecl *D, bool RefersToEnclosingVariableOrCapture, SourceLocation NameLoc, QualType T, ExprValueKind VK, NamedDecl *FoundD=nullptr, const TemplateArgumentListInfo *TemplateArgs=nullptr, NonOdrUseReason NOUR=NOUR_None)
NonOdrUseReason isNonOdrUse() const
Is this expression a non-odr-use reference, and if so, why?
SourceLocation getLocation() const
SourceLocation getLocation() const
bool isUsed(bool CheckUsedAttr=true) const
Whether any (re-)declaration of the entity was used, meaning that a definition is required.
DeclContext * getDeclContext()
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
const Expr * getBase() const
This represents one expression.
const Expr * skipRValueSubobjectAdjustments(SmallVectorImpl< const Expr * > &CommaLHS, SmallVectorImpl< SubobjectAdjustment > &Adjustments) const
Walk outwards from an expression we want to bind a reference to and find the expression whose lifetim...
ExprValueKind getValueKind() const
getValueKind - The value kind that this expression produces.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
bool EvaluateAsRValue(EvalResult &Result, const ASTContext &Ctx, bool InConstantContext=false) const
EvaluateAsRValue - Return true if this is a constant which we can fold to an rvalue using any crazy t...
Decl * getReferencedDeclOfCallee()
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
ExtVectorElementExpr - This represents access to specific elements of a vector, and may occur on the ...
bool isArrow() const
isArrow - Return true if the base expression is a pointer to vector, return false if the base express...
void getEncodedElementAccess(SmallVectorImpl< uint32_t > &Elts) const
getEncodedElementAccess - Encode the elements accessed into an llvm aggregate Constant of ConstantInt...
Represents a member of a struct/union/class.
bool isBitField() const
Determines whether this field is a bitfield.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
unsigned getFieldIndex() const
Returns the index of this field within its record, as appropriate for passing to ASTRecordLayout::get...
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
bool isPotentiallyOverlapping() const
Determine if this field is of potentially-overlapping class type, that is, subobject with the [[no_un...
Represents a function declaration or definition.
FunctionDecl * getPreviousDecl()
Return the previous declaration of this declaration or NULL if this is the first declaration.
Represents a prototype with parameter type info, e.g.
GlobalDecl - represents a global declaration.
const Decl * getDecl() const
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
StorageDuration getStorageDuration() const
Retrieve the storage duration for the materialized temporary.
Expr * getSubExpr() const
Retrieve the temporary-generating subexpression whose value will be materialized into a glvalue.
ValueDecl * getExtendingDecl()
Get the declaration which triggered the lifetime-extension of this temporary, if any.
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
ValueDecl * getMemberDecl() const
Retrieve the member declaration to which this expression refers.
NonOdrUseReason isNonOdrUse() const
Is this expression a non-odr-use reference, and if so, why?
SourceLocation getExprLoc() const LLVM_READONLY
A pointer to member type per C++ 8.3.3 - Pointers to members.
This represents a decl that may have a name.
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
A C++ nested-name-specifier augmented with source location information.
Represents a C++ nested name specifier, such as "\::std::vector<int>::".
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
Expr * getSourceExpr() const
The source expression of an opaque value expression is the expression which originally generated the ...
PointerType - C99 6.7.5.1 - Pointer Declarators.
[C99 6.4.2.2] - A predefined identifier such as func.
StringRef getIdentKindName() const
PredefinedIdentKind getIdentKind() const
StringLiteral * getFunctionName()
A (possibly-)qualified type.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
LangAS getAddressSpace() const
Return the address space of this type.
Qualifiers getQualifiers() const
Retrieve the set of qualifiers applied to this type.
Qualifiers::ObjCLifetime getObjCLifetime() const
Returns lifetime attribute of this type.
QualType withCVRQualifiers(unsigned CVR) const
DestructionKind isDestructedType() const
Returns a nonzero value if objects of this type require non-trivial work to clean up after.
The collection of all-type qualifiers we support.
unsigned getCVRQualifiers() const
@ OCL_ExplicitNone
This object can be modified without requiring retains or releases.
@ OCL_None
There is no lifetime qualification on this type.
void addCVRQualifiers(unsigned mask)
void addQualifiers(Qualifiers Q)
Add the qualifiers from the given set to this set.
Represents a struct/union/class.
Encodes a location in the source.
bool isValid() const
Return true if this is a valid SourceLocation object.
Stmt - This represents one statement.
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
StringLiteral - This represents a string literal expression, e.g.
Exposes information about the current target.
virtual StringRef getABI() const
Get the ABI currently in use.
bool isBooleanType() const
bool isSignedIntegerOrEnumerationType() const
Determines whether this is an integer type that is signed or an enumeration types whose underlying ty...
bool isPackedVectorBoolType(const ASTContext &ctx) const
const ArrayType * castAsArrayTypeUnsafe() const
A variant of castAs<> for array type which silently discards qualifiers from the outermost type.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
bool isFunctionPointerType() const
CXXRecordDecl * castAsCXXRecordDecl() const
bool isArithmeticType() const
bool isConstantMatrixType() const
bool isPointerType() const
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
bool isVariableArrayType() const
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isExtVectorBoolType() const
bool isAnyComplexType() const
const Type * getBaseElementTypeUnsafe() const
Get the base element type of this type, potentially discarding type qualifiers.
bool isAtomicType() const
bool isVariablyModifiedType() const
Whether this type is a variably-modified type (C99 6.7.5).
bool isFunctionType() const
bool isVectorType() const
bool isSubscriptableVectorType() const
const T * getAs() const
Member-template getAs<specific type>'.
bool hasBooleanRepresentation() const
Determine whether this type has a boolean representation – i.e., it is a boolean type,...
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
SourceLocation getExprLoc() const
Expr * getSubExpr() const
static bool isPrefix(Opcode Op)
isPrefix - Return true if this is a prefix operation, like –x.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a variable declaration or definition.
TLSKind getTLSKind() const
VarDecl * getDefinition(ASTContext &)
Get the real (not just tentative) definition for this declaration.
bool hasLocalStorage() const
Returns true if a variable with function scope is a non-static local variable.
@ TLS_Dynamic
TLS with a dynamic initializer.
@ TLS_None
Not a TLS variable.
Represents a C array with a specified size that is not an integer-constant-expression.
Represents a GCC generic vector type.
Defines the clang::TargetInfo interface.
mlir::ptr::MemorySpaceAttrInterface toCIRAddressSpaceAttr(mlir::MLIRContext &ctx, clang::LangAS langAS)
Convert an AST LangAS to the appropriate CIR address space attribute interface.
AlignmentSource
The source of the alignment of an l-value; an expression of confidence in the alignment actually matc...
@ Decl
The l-value was an access to a declared entity or something equivalently strong, like the address of ...
bool isEmptyFieldForLayout(const ASTContext &context, const FieldDecl *fd)
isEmptyFieldForLayout - Return true if the field is "empty", that is, either a zero-width bit-field o...
static AlignmentSource getFieldAlignmentSource(AlignmentSource source)
Given that the base address has the given alignment source, what's our confidence in the alignment of...
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const AstTypeMatcher< FunctionType > functionType
const internal::VariadicDynCastAllOfMatcher< Stmt, CUDAKernelCallExpr > cudaKernelCallExpr
const internal::VariadicDynCastAllOfMatcher< Stmt, Expr > expr
Matches expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CastExpr > castExpr
Matches any cast nodes of Clang's AST.
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
bool isTargetAddressSpace(LangAS AS)
@ SD_Thread
Thread storage duration.
@ SD_Static
Static storage duration.
@ SD_FullExpression
Full-expression storage duration (for temporaries).
@ SD_Automatic
Automatic storage duration (most local variables).
@ SD_Dynamic
Dynamic storage duration.
LangAS
Defines the address space values used by the address space qualifier of QualType.
U cast(CodeGen::Address addr)
@ NOUR_Unevaluated
This name appears in an unevaluated operand.
@ NOUR_Constant
This name appears as a potential result of an lvalue-to-rvalue conversion that is a constant expressi...
static bool weakRefReference()
static bool objCLifetime()
static bool emitLifetimeMarkers()
static bool opLoadEmitScalarRangeCheck()
static bool addressSpace()
static bool opAllocaNonGC()
static bool opAllocaOpenMPThreadPrivate()
static bool preservedAccessIndexRegion()
static bool mergeAllConstants()
static bool opLoadStoreTbaa()
static bool opCallChain()
static bool opAllocaImpreciseLifetime()
static bool opAllocaStaticLocal()
static bool opAllocaTLS()
static bool emitCheckedInBoundsGEP()
static bool attributeNoBuiltin()
static bool setObjCGCLValueClass()
static bool cirgenABIInfo()
static bool opLoadStoreObjC()
static bool opCallArgEvaluationOrder()
static bool insertBuiltinUnpredictable()
static bool opCallMustTail()
static bool shouldReverseUnaryCondOnBoolExpr()
static bool tryEmitAsConstant()
static bool addressIsKnownNonNull()
static bool astVarDeclInterface()
static bool cgCapturedStmtInfo()
static bool opAllocaEscapeByReference()
static bool opLoadStoreNontemporal()
static bool opCallFnInfoOpts()
static bool generateDebugInfo()
static bool incrementProfileCounter()
Record with information about how a bitfield should be accessed.
unsigned volatileStorageSize
The storage size in bits which should be used when accessing this bitfield.
unsigned volatileOffset
The offset within a contiguous run of bitfields that are represented as a single "field" within the c...
std::optional< LValue > rhs
std::optional< LValue > lhs
Represents a scope, including function bodies, compound statements, and the substatements of if/while...
mlir::ptr::MemorySpaceAttrInterface getCIRAllocaAddressSpace() const
EvalResult is a struct with detailed info about an evaluated expression.
APValue Val
Val - This is the value the expression can be folded to.
bool HasSideEffects
Whether the evaluated expression has side effects.