19#include "llvm/Support/SaveAndRestore.h"
30 if (
const auto *CE = dyn_cast_if_present<ConstantExpr>(E);
31 CE && CE->hasAPValueResult() &&
33 return CE->getResultAsAPSInt().getBoolValue();
44 OldInitializingDecl(
Ctx->InitializingDecl) {
45 Ctx->InitializingDecl = VD;
50 this->
Ctx->InitializingDecl = OldInitializingDecl;
51 this->
Ctx->InitStack.pop_back();
64 bool NewInitializing,
bool NewToLValue)
65 : Ctx(Ctx), OldDiscardResult(Ctx->DiscardResult),
66 OldInitializing(Ctx->
Initializing), OldToLValue(Ctx->ToLValue) {
67 Ctx->DiscardResult = NewDiscardResult;
68 Ctx->Initializing = NewInitializing;
69 Ctx->ToLValue = NewToLValue;
73 Ctx->DiscardResult = OldDiscardResult;
74 Ctx->Initializing = OldInitializing;
75 Ctx->ToLValue = OldToLValue;
82 bool OldDiscardResult;
87template <
class Emitter>
91 return Ctx->emitThis(E);
94 return Ctx->emitGetPtrFieldPop(
Offset, E);
96 return Ctx->emitGetPtrLocal(
Offset, E);
100 if (!Ctx->emitConstUint32(
Offset, E))
102 return Ctx->emitArrayElemPtrPopUint32(E);
104 return Ctx->emitRVOPtr(E);
108 llvm_unreachable(
"Unhandled InitLink kind");
124 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
125 assert(LI.Name != Name);
128 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel, ContinueLabel,
149 : Ctx(Ctx), OldCaseLabels(
std::move(this->Ctx->CaseLabels)) {
151 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
152 assert(LI.Name != Name);
155 this->Ctx->CaseLabels = std::move(CaseLabels);
156 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel,
158 DefaultLabel, Ctx->VarScope);
162 this->Ctx->CaseLabels = std::move(OldCaseLabels);
163 this->Ctx->LabelInfoStack.pop_back();
168 CaseMap OldCaseLabels;
174 Ctx->InStmtExpr =
true;
192 : Ctx(Ctx), OldFlag(Ctx->LocOverride), Enabled(Enabled) {
195 Ctx->LocOverride = NewValue;
200 Ctx->LocOverride = OldFlag;
205 std::optional<SourceInfo> OldFlag;
212template <
class Emitter>
220 case CK_LValueToRValue: {
231 if (
const auto *DRE = dyn_cast<DeclRefExpr>(SubExpr)) {
237 if (
auto GlobalIndex =
P.getGlobal(D))
238 return this->emitGetGlobal(*SubExprT, *GlobalIndex, E);
239 }
else if (
auto It =
Locals.find(D); It !=
Locals.end()) {
240 return this->emitGetLocal(*SubExprT, It->second.Offset, E);
241 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
242 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
243 return this->emitGetParam(*SubExprT, It->second.Index, E);
255 if (!this->emitGetPtrLocal(*LocalIndex, E))
259 if (!this->
visit(SubExpr))
263 return this->emitLoadPop(*SubExprT, E);
268 return this->emitMemcpy(E);
271 case CK_DerivedToBaseMemberPointer: {
282 ->getMostRecentCXXRecordDecl();
284 const CXXRecordDecl *ToDecl = B->getType()->getAsCXXRecordDecl();
285 unsigned DerivedOffset =
Ctx.collectBaseOffset(ToDecl, CurDecl);
287 if (!this->emitCastMemberPtrBasePop(DerivedOffset, ToDecl, E))
295 case CK_BaseToDerivedMemberPointer: {
306 ->getMostRecentCXXRecordDecl();
310 typedef std::reverse_iterator<CastExpr::path_const_iterator> ReverseIter;
312 PathI != PathE; ++PathI) {
313 const CXXRecordDecl *ToDecl = (*PathI)->getType()->getAsCXXRecordDecl();
314 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
316 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
323 assert(ToDecl != CurDecl);
324 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
326 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
332 case CK_UncheckedDerivedToBase:
333 case CK_DerivedToBase: {
338 if (
const auto *PT = dyn_cast<PointerType>(Ty))
339 return PT->getPointeeType()->getAsCXXRecordDecl();
340 return Ty->getAsCXXRecordDecl();
347 if (B->isVirtual()) {
348 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), E))
350 CurType = B->getType();
352 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
353 if (!this->emitGetPtrBasePop(
356 CurType = B->getType();
363 case CK_BaseToDerived: {
366 unsigned DerivedOffset =
372 return this->emitGetPtrDerivedPop(DerivedOffset,
377 case CK_FloatingCast: {
382 if (!this->
visit(SubExpr))
384 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
388 case CK_IntegralToFloating: {
391 if (!this->
visit(SubExpr))
393 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
394 return this->emitCastIntegralFloating(
classifyPrim(SubExpr),
395 TargetSemantics, getFPOptions(E), E);
398 case CK_FloatingToBoolean: {
402 if (
const auto *FL = dyn_cast<FloatingLiteral>(SubExpr))
403 return this->emitConstBool(FL->getValue().isNonZero(), E);
404 if (!this->
visit(SubExpr))
406 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
409 case CK_FloatingToIntegral: {
412 if (!this->
visit(SubExpr))
416 return this->emitCastFloatingIntegralAP(
Ctx.getBitWidth(E->
getType()),
419 return this->emitCastFloatingIntegralAPS(
Ctx.getBitWidth(E->
getType()),
422 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
425 case CK_NullToPointer:
426 case CK_NullToMemberPointer: {
431 if (!PointeeType.
isNull()) {
433 Desc =
P.createDescriptor(SubExpr, *T);
435 Desc =
P.createDescriptor(SubExpr, PointeeType.
getTypePtr(),
439 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
443 case CK_PointerToIntegral: {
444 if (!this->
visit(SubExpr))
450 if (!this->emitDecayPtr(FromT,
PT_Ptr, E))
456 return this->emitCastPointerIntegralAP(
Ctx.getBitWidth(E->
getType()), E);
458 return this->emitCastPointerIntegralAPS(
Ctx.getBitWidth(E->
getType()), E);
459 return this->emitCastPointerIntegral(T, E);
462 case CK_ArrayToPointerDecay: {
463 if (!this->
visit(SubExpr))
465 return this->emitArrayDecay(E);
468 case CK_IntegralToPointer: {
470 assert(IntType->isIntegralOrEnumerationType());
471 if (!this->
visit(SubExpr))
479 Desc =
P.createDescriptor(SubExpr, *T);
486 if (!this->emitGetIntPtr(T, Desc, E))
494 return this->emitDecayPtr(
PT_Ptr, DestPtrT, E);
497 case CK_AtomicToNonAtomic:
498 case CK_ConstructorConversion:
499 case CK_FunctionToPointerDecay:
500 case CK_NonAtomicToAtomic:
502 case CK_UserDefinedConversion:
503 case CK_AddressSpaceConversion:
504 case CK_CPointerToObjCPointerCast:
521 return this->emitBuiltinBitCast(E);
536 if (!this->
visit(SubExpr))
544 return this->emitFnPtrCast(E);
551 if (!this->
visit(SubExpr))
553 return this->emitDecayPtr(*FromT, *ToT, E);
555 case CK_IntegralToBoolean:
556 case CK_FixedPointToBoolean: {
562 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr))
563 return this->emitConst(IL->getValue(), E);
564 if (!this->
visit(SubExpr))
569 case CK_BooleanToSignedIntegral:
570 case CK_IntegralCast: {
577 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr)) {
582 if (!this->emitConst(IL->getValue(), SubExpr))
585 if (!this->
visit(SubExpr))
593 if (!ED->isFixed()) {
594 if (!this->emitCheckEnumValue(*FromT, ED, E))
600 if (!this->emitCastAP(*FromT,
Ctx.getBitWidth(E->
getType()), E))
603 if (!this->emitCastAPS(*FromT,
Ctx.getBitWidth(E->
getType()), E))
608 if (!this->emitCast(*FromT, *ToT, E))
611 if (E->
getCastKind() == CK_BooleanToSignedIntegral)
612 return this->emitNeg(*ToT, E);
616 case CK_PointerToBoolean:
617 case CK_MemberPointerToBoolean: {
620 if (!this->
visit(SubExpr))
622 return this->emitIsNonNull(PtrT, E);
625 case CK_IntegralComplexToBoolean:
626 case CK_FloatingComplexToBoolean: {
627 if (!this->
visit(SubExpr))
629 return this->emitComplexBoolCast(SubExpr);
632 case CK_IntegralComplexToReal:
633 case CK_FloatingComplexToReal:
634 return this->emitComplexReal(SubExpr);
636 case CK_IntegralRealToComplex:
637 case CK_FloatingRealToComplex: {
644 if (!this->emitGetPtrLocal(*LocalIndex, E))
653 if (!this->visitZeroInitializer(T, SubExpr->
getType(), SubExpr))
655 return this->emitInitElem(T, 1, SubExpr);
658 case CK_IntegralComplexCast:
659 case CK_FloatingComplexCast:
660 case CK_IntegralComplexToFloatingComplex:
661 case CK_FloatingComplexToIntegralComplex: {
668 if (!this->emitGetPtrLocal(*LocalIndex, E))
675 unsigned SubExprOffset =
677 if (!this->
visit(SubExpr))
679 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
687 for (
unsigned I = 0; I != 2; ++I) {
688 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, E))
690 if (!this->emitArrayElemPop(SourceElemT, I, E))
694 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, E))
698 if (!this->emitInitElem(DestElemT, I, E))
704 case CK_VectorSplat: {
715 if (!this->emitGetPtrLocal(*LocalIndex, E))
721 unsigned ElemOffset =
725 if (!this->
visit(SubExpr))
730 if (!this->emitSetLocal(ElemT, ElemOffset, E))
733 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
734 if (!this->emitGetLocal(ElemT, ElemOffset, E))
736 if (!this->emitInitElem(ElemT, I, E))
743 case CK_HLSLVectorTruncation: {
748 if (!this->
visit(SubExpr))
750 return this->emitArrayElemPop(*ResultT, 0, E);
759 if (!this->emitGetPtrLocal(*LocalIndex, E))
764 if (!this->
visit(SubExpr))
766 return this->emitCopyArray(classifyVectorElementType(E->
getType()), 0, 0,
770 case CK_IntegralToFixedPoint: {
771 if (!this->
visit(SubExpr))
775 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
780 return this->emitPopFixedPoint(E);
783 case CK_FloatingToFixedPoint: {
784 if (!this->
visit(SubExpr))
788 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
789 if (!this->emitCastFloatingFixedPoint(Sem, E))
792 return this->emitPopFixedPoint(E);
795 case CK_FixedPointToFloating: {
796 if (!this->
visit(SubExpr))
798 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
799 if (!this->emitCastFixedPointFloating(TargetSemantics, E))
802 return this->emitPopFloat(E);
805 case CK_FixedPointToIntegral: {
806 if (!this->
visit(SubExpr))
809 if (!this->emitCastFixedPointIntegral(IntegralT, E))
812 return this->emitPop(IntegralT, E);
815 case CK_FixedPointCast: {
816 if (!this->
visit(SubExpr))
819 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
820 if (!this->emitCastFixedPoint(Sem, E))
823 return this->emitPopFixedPoint(E);
835 case CK_LValueBitCast:
838 case CK_HLSLArrayRValue: {
845 if (!this->emitGetPtrLocal(*LocalIndex, E))
848 if (!this->
visit(SubExpr))
850 return this->emitMemcpy(E);
853 case CK_HLSLMatrixTruncation: {
858 if (!this->
visit(SubExpr))
860 return this->emitArrayElemPop(*ResultT, 0, E);
869 if (!this->emitGetPtrLocal(*LocalIndex, E))
874 if (!this->
visit(SubExpr))
876 return this->emitCopyArray(classifyMatrixElementType(SubExpr->
getType()), 0,
880 case CK_HLSLAggregateSplatCast: {
890 if (!this->emitGetPtrLocal(*LocalIndex, E))
900 if (!this->
visit(SubExpr))
902 if (!this->emitSetLocal(SrcElemT, SrcOffset, E))
906 return emitHLSLAggregateSplat(SrcElemT, SrcOffset, E->
getType(), E);
909 case CK_HLSLElementwiseCast: {
920 unsigned SrcPtrOffset =
922 if (!this->
visit(SubExpr))
924 if (!this->emitSetLocal(
PT_Ptr, SrcPtrOffset, E))
928 if (!emitHLSLFlattenAggregate(SrcType, SrcPtrOffset, Elements, 1, E))
930 if (Elements.empty())
933 const HLSLFlatElement &Src = Elements[0];
934 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
936 return this->emitPrimCast(Src.Type, *DestT, DestType, E);
943 if (!this->emitGetPtrLocal(*LocalIndex, E))
949 if (!this->
visit(SubExpr))
951 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
955 unsigned ElemCount = countHLSLFlatElements(DestType);
958 Elements.reserve(ElemCount);
959 if (!emitHLSLFlattenAggregate(SrcType, SrcOffset, Elements, ElemCount, E))
964 assert(Elements.size() == ElemCount &&
965 "Source type has fewer scalar elements than the destination type");
967 return emitHLSLConstructAggregate(DestType, Elements, E);
974 const Record::Field *RF = R->getField(UnionField);
975 QualType FieldType = RF->Decl->getType();
978 if (!this->
visit(SubExpr))
980 if (RF->isBitField())
981 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
983 return this->emitInitFieldActivate(*PT, RF->Offset, E);
986 if (!this->emitGetPtrField(RF->Offset, E))
988 if (!this->emitActivate(E))
994 return this->emitInvalid(E);
996 llvm_unreachable(
"Unhandled clang::CastKind enum");
999template <
class Emitter>
1001 return this->emitBuiltinBitCast(E);
1004template <
class Emitter>
1009 return this->emitConst(
LE->getValue(),
LE);
1012template <
class Emitter>
1018 return this->emitFloat(F, E);
1021template <
class Emitter>
1031 if (!this->emitGetPtrLocal(*LocalIndex, E))
1038 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
1040 if (!this->emitInitElem(SubExprT, 0, SubExpr))
1045template <
class Emitter>
1053 auto Sem =
Ctx.getASTContext().getFixedPointSemantics(E->
getType());
1058template <
class Emitter>
1063template <
class Emitter>
1090 return this->emitComplexComparison(LHS, RHS, E);
1098 if (!this->
visit(LHS))
1101 if (!this->
visit(RHS))
1104 if (!this->emitToMemberPtr(E))
1110 if (!this->emitCastMemberPtrPtr(E))
1127 Ctx.getASTContext().CompCategories.lookupInfoForType(E->
getType());
1133 if (!this->emitGetPtrLocal(*ResultIndex, E))
1140 return this->emitCMP3(*
LT, CmpInfo, E);
1143 if (!
LT || !RT || !T)
1153 return this->visitAssignment(LHS, RHS, E);
1160 auto MaybeCastToBool = [
this, T, E](
bool Result) {
1164 return this->emitPopBool(E);
1166 return this->emitCast(
PT_Bool, *T, E);
1170 auto Discard = [
this, T, E](
bool Result) {
1178 return MaybeCastToBool(this->emitEQ(*
LT, E));
1180 return MaybeCastToBool(this->emitNE(*
LT, E));
1182 return MaybeCastToBool(this->emitLT(*
LT, E));
1184 return MaybeCastToBool(this->emitLE(*
LT, E));
1186 return MaybeCastToBool(this->emitGT(*
LT, E));
1188 return MaybeCastToBool(this->emitGE(*
LT, E));
1191 return Discard(this->emitSubf(getFPOptions(E), E));
1192 return Discard(this->emitSub(*T, E));
1195 return Discard(this->emitAddf(getFPOptions(E), E));
1196 return Discard(this->emitAdd(*T, E));
1199 return Discard(this->emitMulf(getFPOptions(E), E));
1200 return Discard(this->emitMul(*T, E));
1202 return Discard(this->emitRem(*T, E));
1205 return Discard(this->emitDivf(getFPOptions(E), E));
1206 return Discard(this->emitDiv(*T, E));
1208 return Discard(this->emitBitAnd(*T, E));
1210 return Discard(this->emitBitOr(*T, E));
1212 return Discard(this->emitShl(*
LT, *RT, E));
1214 return Discard(this->emitShr(*
LT, *RT, E));
1216 return Discard(this->emitBitXor(*T, E));
1219 llvm_unreachable(
"Already handled earlier");
1224 llvm_unreachable(
"Unhandled binary op");
1229template <
class Emitter>
1235 if ((Op != BO_Add && Op != BO_Sub) ||
1246 auto visitAsPointer = [&](
const Expr *E,
PrimType T) ->
bool {
1247 if (!this->
visit(E))
1250 return this->emitDecayPtr(T,
PT_Ptr, E);
1259 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1267 ElemTypeSize =
Ctx.getASTContext().getTypeSizeInChars(ElemType);
1270 if (!this->emitSubPtr(IntT, ElemTypeSize.
isZero(), E))
1277 if (!visitAsPointer(RHS, *RT))
1279 if (!this->
visit(LHS))
1283 if (!visitAsPointer(LHS, *
LT))
1285 if (!this->
visit(RHS))
1296 if (!this->emitAddOffset(OffsetType, E))
1300 if (!this->emitSubOffset(OffsetType, E))
1317template <
class Emitter>
1327 LabelTy LabelTrue = this->getLabel();
1328 LabelTy LabelEnd = this->getLabel();
1332 if (!this->jumpTrue(LabelTrue, E))
1337 if (!this->jump(LabelEnd, E))
1340 this->emitLabel(LabelTrue);
1341 this->emitConstBool(
true, E);
1342 this->fallthrough(LabelEnd);
1343 this->emitLabel(LabelEnd);
1346 assert(Op == BO_LAnd);
1349 LabelTy LabelFalse = this->getLabel();
1350 LabelTy LabelEnd = this->getLabel();
1354 if (!this->jumpFalse(LabelFalse, E))
1359 if (!this->jump(LabelEnd, E))
1362 this->emitLabel(LabelFalse);
1363 this->emitConstBool(
false, E);
1364 this->fallthrough(LabelEnd);
1365 this->emitLabel(LabelEnd);
1369 return this->emitPopBool(E);
1374 return this->emitCast(
PT_Bool, *T, E);
1378template <
class Emitter>
1385 if (!this->emitGetPtrLocal(*LocalIndex, E))
1394 PrimType ResultElemT = this->classifyComplexElementType(E->
getType());
1395 unsigned ResultOffset = ~0u;
1401 if (!this->emitDupPtr(E))
1403 if (!this->emitSetLocal(
PT_Ptr, ResultOffset, E))
1408 LHSType = AT->getValueType();
1411 RHSType = AT->getValueType();
1420 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1425 if (!this->
visit(LHS))
1427 if (!this->
visit(RHS))
1429 if (!this->emitMulc(ElemT, E))
1432 return this->emitPopPtr(E);
1436 if (Op == BO_Div && RHSIsComplex) {
1443 if (!LHSIsComplex) {
1448 LHSOffset = *LocalIndex;
1450 if (!this->emitGetPtrLocal(LHSOffset, E))
1453 if (!this->
visit(LHS))
1456 if (!this->emitInitElem(ElemT, 0, E))
1459 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1461 if (!this->emitInitElem(ElemT, 1, E))
1464 if (!this->
visit(LHS))
1468 if (!this->
visit(RHS))
1470 if (!this->emitDivc(ElemT, E))
1473 return this->emitPopPtr(E);
1480 if (!this->
visit(LHS))
1482 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1487 if (!this->
visit(LHS))
1489 if (!this->emitSetLocal(LHST, LHSOffset, E))
1497 if (!this->
visit(RHS))
1499 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1504 if (!this->
visit(RHS))
1506 if (!this->emitSetLocal(RHST, RHSOffset, E))
1513 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1514 unsigned ElemIndex,
unsigned Offset,
1515 const Expr *E) ->
bool {
1517 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1519 return this->emitArrayElemPop(classifyComplexElementType(E->
getType()),
1522 if (ElemIndex == 0 || !LoadZero)
1529 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1532 if (!this->emitGetLocal(
PT_Ptr, ResultOffset, E))
1539 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1542 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1545 if (!this->emitAddf(getFPOptions(E), E))
1548 if (!this->emitAdd(ResultElemT, E))
1553 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1556 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1559 if (!this->emitSubf(getFPOptions(E), E))
1562 if (!this->emitSub(ResultElemT, E))
1567 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1570 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1574 if (!this->emitMulf(getFPOptions(E), E))
1577 if (!this->emitMul(ResultElemT, E))
1582 assert(!RHSIsComplex);
1583 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1586 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1590 if (!this->emitDivf(getFPOptions(E), E))
1593 if (!this->emitDiv(ResultElemT, E))
1604 if (!this->emitInitElemPop(ResultElemT, ElemIndex, E))
1607 if (!this->emitPop(ResultElemT, E))
1612 return this->emitPopPtr(E);
1618template <
class Emitter>
1623 "Comma op should be handled in VisitBinaryOperator");
1637 if (!this->emitGetPtrLocal(*LocalIndex, E))
1651 assert(
Ctx.getASTContext().hasSameUnqualifiedType(
1654 if (!this->
visit(LHS))
1656 if (!this->
visit(RHS))
1658 if (!this->emitCopyArray(ElemT, 0, 0, VecTy->getNumElements(), E))
1661 return this->emitPopPtr(E);
1666 unsigned LHSOffset =
1668 if (!this->
visit(LHS))
1670 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1674 unsigned RHSOffset =
1676 if (!this->
visit(RHS))
1678 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1690 if (NeedIntPromot) {
1692 Ctx.getASTContext().getPromotedIntegerType(
Ctx.getASTContext().BoolTy);
1697 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1698 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1700 if (!this->emitArrayElemPop(ElemT, Index, E))
1703 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
1705 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1707 }
else if (NeedIntPromot) {
1708 if (!this->emitPrimCast(ElemT, PromotT, PromotTy, E))
1714#define EMIT_ARITH_OP(OP) \
1716 if (ElemT == PT_Float) { \
1717 if (!this->emit##OP##f(getFPOptions(E), E)) \
1720 if (!this->emit##OP(ElemT, E)) \
1726 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1727 if (!getElem(LHSOffset, ElemT, I))
1729 if (!getElem(RHSOffset, RHSElemT, I))
1741 if (!this->emitRem(ElemT, E))
1745 if (!this->emitBitAnd(OpT, E))
1749 if (!this->emitBitOr(OpT, E))
1753 if (!this->emitBitXor(OpT, E))
1757 if (!this->emitShl(OpT, RHSElemT, E))
1761 if (!this->emitShr(OpT, RHSElemT, E))
1765 if (!this->emitEQ(ElemT, E))
1769 if (!this->emitNE(ElemT, E))
1773 if (!this->emitLE(ElemT, E))
1777 if (!this->emitLT(ElemT, E))
1781 if (!this->emitGE(ElemT, E))
1785 if (!this->emitGT(ElemT, E))
1790 if (!this->emitBitAnd(ResultElemT, E))
1795 if (!this->emitBitOr(ResultElemT, E))
1799 return this->emitInvalid(E);
1808 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1810 if (!this->emitNeg(ResultElemT, E))
1816 if (NeedIntPromot &&
1817 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(), E))
1821 if (!this->emitInitElem(ResultElemT, I, E))
1830template <
class Emitter>
1840 auto LHSSemaInt = LHSSema.toOpaqueInt();
1842 auto RHSSemaInt = RHSSema.toOpaqueInt();
1844 if (!this->
visit(LHS))
1852 if (!this->
visit(RHS))
1861 auto ConvertResult = [&](
bool R) ->
bool {
1865 auto CommonSema = LHSSema.getCommonSemantics(RHSSema).toOpaqueInt();
1866 if (ResultSema != CommonSema)
1867 return this->emitCastFixedPoint(ResultSema, E);
1871 auto MaybeCastToBool = [&](
bool Result) {
1876 return this->emitPop(T, E);
1878 return this->emitCast(
PT_Bool, T, E);
1884 return MaybeCastToBool(this->emitEQFixedPoint(E));
1886 return MaybeCastToBool(this->emitNEFixedPoint(E));
1888 return MaybeCastToBool(this->emitLTFixedPoint(E));
1890 return MaybeCastToBool(this->emitLEFixedPoint(E));
1892 return MaybeCastToBool(this->emitGTFixedPoint(E));
1894 return MaybeCastToBool(this->emitGEFixedPoint(E));
1896 return ConvertResult(this->emitAddFixedPoint(E));
1898 return ConvertResult(this->emitSubFixedPoint(E));
1900 return ConvertResult(this->emitMulFixedPoint(E));
1902 return ConvertResult(this->emitDivFixedPoint(E));
1904 return ConvertResult(this->emitShiftFixedPoint(
true, E));
1906 return ConvertResult(this->emitShiftFixedPoint(
false, E));
1909 return this->emitInvalid(E);
1912 llvm_unreachable(
"unhandled binop opcode");
1915template <
class Emitter>
1924 if (!this->
visit(SubExpr))
1926 if (!this->emitNegFixedPoint(E))
1929 return this->emitPopFixedPoint(E);
1935 llvm_unreachable(
"Unhandled unary opcode");
1938template <
class Emitter>
1947 return this->visitZeroInitializer(*T, QT, E);
1955 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1956 CXXRD && CXXRD->getNumVBases() > 0) {
1966 return this->visitZeroRecordInitializer(R, E);
1973 return this->visitZeroArrayInitializer(QT, E);
1977 QualType ElemQT = ComplexTy->getElementType();
1979 for (
unsigned I = 0; I < 2; ++I) {
1980 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1982 if (!this->emitInitElem(ElemT, I, E))
1989 unsigned NumVecElements = VecT->getNumElements();
1990 QualType ElemQT = VecT->getElementType();
1993 for (
unsigned I = 0; I < NumVecElements; ++I) {
1994 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1996 if (!this->emitInitElem(ElemT, I, E))
2003 unsigned NumElems = MT->getNumElementsFlattened();
2004 QualType ElemQT = MT->getElementType();
2007 for (
unsigned I = 0; I != NumElems; ++I) {
2008 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2010 if (!this->emitInitElem(ElemT, I, E))
2019template <
class Emitter>
2032 for (
const Expr *SubExpr : {LHS, RHS}) {
2033 if (!this->
visit(SubExpr)) {
2040 if (SubExpr ==
Base &&
Base->getType()->isPointerType()) {
2041 if (!this->emitExpandPtr(E))
2052 return this->emitError(E);
2055 if (!this->emitFlip(
PT_Ptr, *IndexT, E))
2059 if (!this->emitArrayElemPtrPop(*IndexT, E))
2062 return this->emitPopPtr(E);
2068 return this->emitLoadPop(*T, E);
2071template <
class Emitter>
2073 const Expr *ArrayFiller,
const Expr *E) {
2078 QT = AT->getValueType();
2081 if (
Inits.size() == 0)
2083 return this->emitInvalid(E);
2098 if (
Inits.size() == 0)
2099 return this->visitZeroInitializer(*T, QT, E);
2100 assert(
Inits.size() == 1);
2113 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
2120 bool BitField = FieldToInit->isBitField();
2122 return this->emitInitBitFieldActivate(T, FieldToInit->Offset,
2123 FieldToInit->bitWidth(), E);
2125 return this->emitInitBitField(T, FieldToInit->Offset,
2126 FieldToInit->bitWidth(), E);
2128 return this->emitInitFieldActivate(T, FieldToInit->Offset, E);
2129 return this->emitInitField(T, FieldToInit->Offset, E);
2132 auto initCompositeField = [=](
const Record::Field *FieldToInit,
2140 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
2143 if (
Activate && !this->emitActivate(E))
2150 if (
Inits.size() == 0) {
2151 if (!this->visitZeroRecordInitializer(R, E))
2156 if (
const auto *ILE = dyn_cast<InitListExpr>(E))
2157 FToInit = ILE->getInitializedFieldInUnion();
2161 const Record::Field *FieldToInit = R->getField(FToInit);
2163 if (!initPrimitiveField(FieldToInit,
Init, *T,
true))
2166 if (!initCompositeField(FieldToInit,
Init,
true))
2170 return this->emitFinishInit(E);
2173 assert(!R->isUnion());
2174 unsigned InitIndex = 0;
2177 while (InitIndex < R->getNumFields() &&
2178 R->getField(InitIndex)->isUnnamedBitField())
2182 const Record::Field *FieldToInit = R->getField(InitIndex);
2183 if (!initPrimitiveField(FieldToInit,
Init, *T))
2188 if (
const Record::Base *B = R->getBase(
Init->getType())) {
2189 if (!this->emitGetPtrBase(B->Offset,
Init))
2197 const Record::Field *FieldToInit = R->getField(InitIndex);
2198 if (!initCompositeField(FieldToInit,
Init))
2204 return this->emitFinishInit(E);
2208 if (
Inits.size() == 1 && QT ==
Inits[0]->getType())
2212 Ctx.getASTContext().getAsConstantArrayType(QT);
2215 if (!this->emitCheckArraySize(NumElems, E))
2219 unsigned ElementIndex = 0;
2221 if (
const auto *EmbedS =
2222 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
2230 if (!this->emitCastIntegralFloating(
classifyPrim(IL), Sem,
2231 getFPOptions(E), E))
2237 return this->emitInitElem(TargetT, ElemIndex, IL);
2239 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
2251 for (; ElementIndex != NumElems; ++ElementIndex) {
2257 return this->emitFinishInit(E);
2261 unsigned NumInits =
Inits.size();
2266 QualType ElemQT = ComplexTy->getElementType();
2268 if (NumInits == 0) {
2270 for (
unsigned I = 0; I < 2; ++I) {
2271 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2273 if (!this->emitInitElem(ElemT, I, E))
2276 }
else if (NumInits == 2) {
2277 unsigned InitIndex = 0;
2282 if (!this->emitInitElem(ElemT, InitIndex, E))
2291 unsigned NumVecElements = VecT->getNumElements();
2292 assert(NumVecElements >=
Inits.size());
2294 QualType ElemQT = VecT->getElementType();
2298 unsigned InitIndex = 0;
2305 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
2306 if (!this->emitCopyArray(ElemT, 0, InitIndex,
2307 InitVecT->getNumElements(), E))
2309 InitIndex += InitVecT->getNumElements();
2311 if (!this->emitInitElem(ElemT, InitIndex, E))
2317 assert(InitIndex <= NumVecElements);
2320 for (; InitIndex != NumVecElements; ++InitIndex) {
2321 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2323 if (!this->emitInitElem(ElemT, InitIndex, E))
2330 unsigned NumElems = MT->getNumElementsFlattened();
2331 assert(
Inits.size() == NumElems);
2333 QualType ElemQT = MT->getElementType();
2339 for (
unsigned I = 0; I != NumElems; ++I) {
2342 if (!this->emitInitElem(ElemT, I, E))
2353template <
class Emitter>
2360 return this->emitInitElem(*InitT, ElemIndex,
Init);
2366 if (!this->emitConstUint32(ElemIndex,
Init))
2368 if (!this->emitArrayElemPtrUint32(
Init))
2373template <
class Emitter>
2376 bool Activate,
bool IsOperatorCall) {
2378 llvm::BitVector NonNullArgs;
2379 if (FuncDecl && FuncDecl->
hasAttr<NonNullAttr>())
2382 bool ExplicitMemberFn =
false;
2383 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl))
2384 ExplicitMemberFn = MD->isExplicitObjectMemberFunction();
2386 unsigned ArgIndex = 0;
2387 for (
const Expr *Arg : Args) {
2389 if (!this->
visit(Arg))
2397 unsigned DeclIndex = ArgIndex - IsOperatorCall + ExplicitMemberFn;
2398 if (DeclIndex < FuncDecl->getNumParams())
2399 Source = FuncDecl->
getParamDecl(ArgIndex - IsOperatorCall +
2408 if (!this->emitGetPtrLocal(*LocalIndex, Arg))
2416 if (!this->emitActivate(Arg))
2420 if (!NonNullArgs.empty() && NonNullArgs[ArgIndex]) {
2423 if (!this->emitCheckNonNullArg(ArgT, Arg))
2434template <
class Emitter>
2439template <
class Emitter>
2445template <
class Emitter>
2451template <
class Emitter>
2469template <
class Emitter>
2471 auto It = E->
begin();
2472 return this->
visit(*It);
2477 bool AlignOfReturnsPreferred =
2478 ASTCtx.
getLangOpts().getClangABICompat() <= LangOptions::ClangABI::Ver7;
2484 T = Ref->getPointeeType();
2486 if (T.getQualifiers().hasUnaligned())
2492 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2498template <
class Emitter>
2505 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2511 ArgType = Ref->getPointeeType();
2517 if (
ArgType->isDependentType() || !
ArgType->isConstantSizeType())
2518 return this->emitInvalid(E);
2520 if (Kind == UETT_SizeOf)
2529 return this->emitConst(Size.getQuantity(), E);
2532 if (Kind == UETT_CountOf) {
2538 if (
const auto *CAT =
2542 return this->emitConst(CAT->getSize(), E);
2552 if (VAT->getElementType()->isArrayType()) {
2553 std::optional<APSInt> Res =
2555 ? VAT->getSizeExpr()->getIntegerConstantExpr(ASTCtx)
2560 return this->emitConst(*Res, E);
2565 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2585 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2588 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2598 return this->emitConst(Size.getQuantity(), E);
2601 if (Kind == UETT_VectorElements) {
2606 return this->emitConst(VT->getNumElements(), E);
2608 return this->emitSizelessVectorElementSize(E);
2611 if (Kind == UETT_VecStep) {
2613 unsigned N = VT->getNumElements();
2620 return this->emitConst(N, E);
2622 return this->emitConst(1, E);
2625 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2632 if (Kind == UETT_PtrAuthTypeDiscriminator) {
2634 return this->emitInvalid(E);
2636 return this->emitConst(
2637 const_cast<ASTContext &
>(ASTCtx).getPointerAuthTypeDiscriminator(
2645template <
class Emitter>
2654 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2657 if (
auto GlobalIndex =
P.getGlobal(VD)) {
2658 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
2660 if (
Member->getType()->isReferenceType())
2661 return this->emitLoadPopPtr(E);
2668 if (!this->
discard(Base) && !this->emitSideEffect(E))
2683 const Record::Field *F = R->getField(FD);
2687 const auto maybeLoadValue = [&]() ->
bool {
2691 return this->emitLoadPop(*T, E);
2696 if (F->Decl->getType()->isReferenceType())
2697 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2698 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2701template <
class Emitter>
2711template <
class Emitter>
2725 if (!this->
visit(Common))
2727 return this->emitCopyArray(*SubExprT, 0, 0, Size, E);
2741 for (
size_t I = 0; I != Size; ++I) {
2753template <
class Emitter>
2768 return this->emitGetLocal(SubExprT, It->second, E);
2771 if (!this->
visit(SourceExpr))
2778 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2787 return this->emitGetLocal(SubExprT, LocalIndex, E);
2791template <
class Emitter>
2814 bool IsBcpCall =
false;
2815 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2816 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2820 LabelTy LabelEnd = this->getLabel();
2821 LabelTy LabelFalse = this->getLabel();
2824 if (!this->emitPushIgnoreDiags(E))
2832 if (this->checkingForUndefinedBehavior()) {
2835 if (!this->
discard(FalseExpr))
2841 if (!this->jumpFalse(LabelFalse, E))
2846 if (!this->jump(LabelEnd, E))
2848 this->emitLabel(LabelFalse);
2852 this->fallthrough(LabelEnd);
2853 this->emitLabel(LabelEnd);
2856 return this->emitPopIgnoreDiags(E);
2860template <
class Emitter>
2866 unsigned StringIndex =
P.createGlobalString(E);
2867 return this->emitGetPtrGlobal(StringIndex, E);
2872 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2873 assert(CAT &&
"a string literal that's not a constant array?");
2878 unsigned N = std::min(ArraySize, E->
getLength());
2881 for (
unsigned I = 0; I != N; ++I) {
2884 if (CharWidth == 1) {
2885 this->emitConstSint8(CodeUnit, E);
2886 this->emitInitElemSint8(I, E);
2887 }
else if (CharWidth == 2) {
2888 this->emitConstUint16(CodeUnit, E);
2889 this->emitInitElemUint16(I, E);
2890 }
else if (CharWidth == 4) {
2891 this->emitConstUint32(CodeUnit, E);
2892 this->emitInitElemUint32(I, E);
2894 llvm_unreachable(
"unsupported character width");
2899 for (
unsigned I = N; I != ArraySize; ++I) {
2900 if (CharWidth == 1) {
2901 this->emitConstSint8(0, E);
2902 this->emitInitElemSint8(I, E);
2903 }
else if (CharWidth == 2) {
2904 this->emitConstUint16(0, E);
2905 this->emitInitElemUint16(I, E);
2906 }
else if (CharWidth == 4) {
2907 this->emitConstUint32(0, E);
2908 this->emitInitElemUint32(I, E);
2910 llvm_unreachable(
"unsupported character width");
2917template <
class Emitter>
2921 return this->emitDummyPtr(E, E);
2924template <
class Emitter>
2926 auto &A =
Ctx.getASTContext();
2935template <
class Emitter>
2943 auto &A =
Ctx.getASTContext();
2947 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2948 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2955 unsigned StringIndex =
P.createGlobalString(SL);
2956 return this->emitGetPtrGlobal(StringIndex, E);
2959template <
class Emitter>
2963 return this->emitConst(E->
getValue(), E);
2966template <
class Emitter>
2992 if (!this->emitSetLocal(*RT, TempOffset, E))
2998 if (!this->emitLoad(LHST, E))
3002 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
3003 LHSComputationType, E))
3007 if (!this->emitGetLocal(*RT, TempOffset, E))
3012 if (!this->emitAddf(getFPOptions(E), E))
3016 if (!this->emitSubf(getFPOptions(E), E))
3020 if (!this->emitMulf(getFPOptions(E), E))
3024 if (!this->emitDivf(getFPOptions(E), E))
3035 return this->emitStorePop(LHST, E);
3036 return this->emitStore(LHST, E);
3039template <
class Emitter>
3048 if (Op != BO_AddAssign && Op != BO_SubAssign)
3057 if (!this->emitLoad(*
LT, LHS))
3063 if (Op == BO_AddAssign) {
3064 if (!this->emitAddOffset(*RT, E))
3067 if (!this->emitSubOffset(*RT, E))
3072 return this->emitStorePopPtr(E);
3073 return this->emitStorePtr(E);
3076template <
class Emitter>
3089 if (!
Ctx.getLangOpts().CPlusPlus14)
3090 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
3092 if (!
LT || !RT || !ResultT || !LHSComputationT)
3117 if (!this->emitSetLocal(*RT, TempOffset, E))
3124 if (!this->emitLoad(*
LT, E))
3126 if (
LT != LHSComputationT &&
3132 if (!this->emitGetLocal(*RT, TempOffset, E))
3138 if (!this->emitAdd(*LHSComputationT, E))
3142 if (!this->emitSub(*LHSComputationT, E))
3146 if (!this->emitMul(*LHSComputationT, E))
3150 if (!this->emitDiv(*LHSComputationT, E))
3154 if (!this->emitRem(*LHSComputationT, E))
3158 if (!this->emitShl(*LHSComputationT, *RT, E))
3162 if (!this->emitShr(*LHSComputationT, *RT, E))
3166 if (!this->emitBitAnd(*LHSComputationT, E))
3170 if (!this->emitBitXor(*LHSComputationT, E))
3174 if (!this->emitBitOr(*LHSComputationT, E))
3178 llvm_unreachable(
"Unimplemented compound assign operator");
3182 if (ResultT != LHSComputationT &&
3183 !this->emitIntegralCast(*LHSComputationT, *ResultT, E->
getType(), E))
3189 return this->emitStoreBitFieldPop(*ResultT, E);
3190 return this->emitStorePop(*ResultT, E);
3193 return this->emitStoreBitField(*ResultT, E);
3194 return this->emitStore(*ResultT, E);
3197template <
class Emitter>
3205template <
class Emitter>
3220 if (!
Ctx.getLangOpts().CPlusPlus11)
3227 for (
const Expr *LHS : CommaLHSs) {
3247 if (!this->
visit(Inner))
3249 if (!this->emitInitGlobalTemp(*InnerT, *GlobalIndex, TempDecl, E))
3251 return this->emitGetPtrGlobal(*GlobalIndex, E);
3254 if (!this->checkLiteralType(Inner))
3257 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3261 return this->emitInitGlobalTempComp(TempDecl, E);
3272 unsigned LocalIndex =
3274 if (!this->VarScope->LocalsAlwaysEnabled &&
3275 !this->emitEnableLocal(LocalIndex, E))
3278 if (!this->
visit(Inner))
3280 if (!this->emitSetLocal(*InnerT, LocalIndex, E))
3283 return this->emitGetPtrLocal(LocalIndex, E);
3286 if (!this->checkLiteralType(Inner))
3293 if (!this->VarScope->LocalsAlwaysEnabled &&
3294 !this->emitEnableLocal(*LocalIndex, E))
3297 if (!this->emitGetPtrLocal(*LocalIndex, E))
3304template <
class Emitter>
3315 if (!this->
visit(SubExpr))
3319 return this->emitPopPtr(E);
3323template <
class Emitter>
3344 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3349 if (
P.isGlobalInitialized(*GlobalIndex))
3355 return this->emitInitGlobal(*T, *GlobalIndex, E);
3367 unsigned LocalIndex;
3371 LocalIndex = *MaybeIndex;
3375 if (!this->emitGetPtrLocal(LocalIndex, E))
3379 return this->
visit(
Init) && this->emitInit(*T, E);
3383template <
class Emitter>
3396template <
class Emitter>
3400 return this->emitConst(E->
getValue(), E);
3403template <
class Emitter>
3416 for (
const Record::Field &F : R->fields()) {
3418 if (!
Init ||
Init->containsErrors())
3426 if (!this->emitInitField(*T, F.Offset, E))
3429 if (!this->emitGetPtrField(F.Offset, E))
3440template <
class Emitter>
3447 return this->emitGetPtrGlobal(StringIndex, E);
3453template <
class Emitter>
3458 return this->emitInvalid(E);
3461template <
class Emitter>
3487 if (PointeeToT && PointeeFromT) {
3506 bool Fatal = (ToT != FromT);
3513template <
class Emitter>
3516 if (!
Ctx.getLangOpts().CPlusPlus20) {
3524template <
class Emitter>
3530 return this->emitConstBool(E->
getValue(), E);
3533template <
class Emitter>
3538 if (T->isRecordType()) {
3552 if (!this->emitGetPtrLocal(*LocalIndex, E))
3560 T->getAsCXXRecordDecl()))
3570 if (!this->visitZeroRecordInitializer(R, E))
3582 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3584 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3585 if (!this->emitCheckFunctionDecl(Ctor, E))
3596 assert(
Func->hasThisPointer());
3597 assert(!
Func->hasRVO());
3601 if (!this->emitDupPtr(E))
3605 for (
const auto *Arg : E->
arguments()) {
3606 if (!this->
visit(Arg))
3610 if (
Func->isVariadic()) {
3611 uint32_t VarArgSize = 0;
3612 unsigned NumParams =
Func->getNumWrittenParams();
3613 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3617 if (!this->emitCallVar(
Func, VarArgSize, E))
3620 if (!this->emitCall(
Func, 0, E)) {
3625 (void)this->emitPopPtr(E);
3631 return this->emitPopPtr(E);
3635 if (T->isArrayType()) {
3640 if (!this->emitDupPtr(E))
3644 initArrayDimension = [&](
QualType T) ->
bool {
3645 if (!T->isArrayType()) {
3647 for (
const auto *Arg : E->
arguments()) {
3648 if (!this->
visit(Arg))
3652 return this->emitCall(
Func, 0, E);
3656 Ctx.getASTContext().getAsConstantArrayType(T);
3661 for (
size_t I = 0; I != NumElems; ++I) {
3662 if (!this->emitConstUint64(I, E))
3664 if (!this->emitArrayElemPtrUint64(E))
3666 if (!initArrayDimension(ElemTy))
3669 return this->emitPopPtr(E);
3672 return initArrayDimension(E->
getType());
3678template <
class Emitter>
3688 assert(Val.
isInt());
3690 return this->emitConst(I, E);
3697 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3698 return this->
visit(LValueExpr);
3713 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3717 const APValue &
V = UGCD->getValue();
3718 for (
unsigned I = 0, N = R->getNumFields(); I != N; ++I) {
3719 const Record::Field *F = R->getField(I);
3720 const APValue &FieldValue =
V.getStructField(I);
3726 if (!this->emitInitField(FieldT, F->Offset, E))
3734template <
class Emitter>
3740 for (
unsigned I = 0; I != N; ++I) {
3747 if (!this->
discard(ArrayIndexExpr))
3752 if (!this->
visit(ArrayIndexExpr))
3756 if (!this->emitCast(IndexT,
PT_Sint64, E))
3766 return this->emitOffsetOf(T, E, E);
3769template <
class Emitter>
3778 return this->visitZeroInitializer(*T, Ty, E);
3785 if (!this->emitGetPtrLocal(*LocalIndex, E))
3790 QualType ElemQT = CT->getElementType();
3793 for (
unsigned I = 0; I != 2; ++I) {
3794 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3796 if (!this->emitInitElem(ElemT, I, E))
3808 if (!this->emitGetPtrLocal(*LocalIndex, E))
3813 QualType ElemQT = VT->getElementType();
3816 for (
unsigned I = 0, N = VT->getNumElements(); I != N; ++I) {
3817 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3819 if (!this->emitInitElem(ElemT, I, E))
3828template <
class Emitter>
3833template <
class Emitter>
3839template <
class Emitter>
3844template <
class Emitter>
3849 return this->emitConst(E->
getValue(), E);
3852template <
class Emitter>
3857 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3871 unsigned ParamIndex = 0;
3875 if (!this->emitGetParam(PT, ParamIndex, E))
3880 return this->emitCall(F, 0, E);
3885template <
class Emitter>
3893 const Expr *PlacementDest =
nullptr;
3894 bool IsNoThrow =
false;
3899 if (PlacementArgs != 0) {
3908 if (PlacementArgs == 1) {
3916 if (!this->emitInvalidNewDeleteExpr(E, E))
3921 if (OperatorNew->isReservedGlobalPlacementOperator())
3922 PlacementDest = Arg1;
3926 return this->emitInvalid(E);
3928 }
else if (!OperatorNew
3929 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3930 return this->emitInvalidNewDeleteExpr(E, E);
3933 if (!PlacementDest) {
3938 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3941 Desc =
P.createDescriptor(
3944 false,
false,
false,
3950 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3954 const Expr *Stripped = *ArraySizeExpr;
3955 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3956 Stripped = ICE->getSubExpr())
3957 if (ICE->getCastKind() != CK_NoOp &&
3958 ICE->getCastKind() != CK_IntegralCast)
3966 if (!this->
visit(Stripped))
3968 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
3971 if (PlacementDest) {
3972 if (!this->
visit(PlacementDest))
3974 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3976 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
3979 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3984 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
3988 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
3995 size_t StaticInitElems = 0;
3996 const Expr *DynamicInit =
nullptr;
4000 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
4001 StaticInitElems = CAT->getZExtSize();
4006 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init)) {
4007 if (ILE->hasArrayFiller())
4008 DynamicInit = ILE->getArrayFiller();
4027 const Function *CtorFunc =
nullptr;
4028 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
4032 }
else if (!DynamicInit && !ElemT)
4035 LabelTy EndLabel = this->getLabel();
4036 LabelTy StartLabel = this->getLabel();
4041 if (!this->emitDupPtr(E))
4043 if (!this->emitNullPtr(0,
nullptr, E))
4045 if (!this->emitEQPtr(E))
4047 if (!this->jumpTrue(EndLabel, E))
4054 if (!this->emitConst(StaticInitElems,
SizeT, E))
4056 if (!this->emitSetLocal(
SizeT, Iter, E))
4059 this->fallthrough(StartLabel);
4060 this->emitLabel(StartLabel);
4062 if (!this->emitGetLocal(
SizeT, Iter, E))
4064 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4066 if (!this->emitLT(
SizeT, E))
4068 if (!this->jumpFalse(EndLabel, E))
4072 if (!this->emitGetLocal(
SizeT, Iter, E))
4074 if (!this->emitArrayElemPtr(
SizeT, E))
4077 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
4082 if (!this->visitZeroInitializer(InitT, ElemType, E))
4084 if (!this->emitStorePop(InitT, E))
4086 }
else if (DynamicInit) {
4088 if (!this->
visit(DynamicInit))
4090 if (!this->emitStorePop(*InitT, E))
4097 if (!this->visitZeroInitializer(
4101 if (!this->emitStorePop(*ElemT, E))
4105 if (!this->emitCall(CtorFunc, 0, E))
4110 if (!this->emitGetPtrLocal(Iter, E))
4112 if (!this->emitIncPop(
SizeT,
false, E))
4115 if (!this->jump(StartLabel, E))
4118 this->fallthrough(EndLabel);
4119 this->emitLabel(EndLabel);
4123 if (PlacementDest) {
4124 if (!this->
visit(PlacementDest))
4126 if (!this->emitCheckNewTypeMismatch(E, E))
4131 if (!this->emitAlloc(Desc, E))
4140 if (!this->emitInit(*ElemT, E))
4151 return this->emitPopPtr(E);
4156template <
class Emitter>
4162 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
4163 return this->emitInvalidNewDeleteExpr(E, E);
4172template <
class Emitter>
4178 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
4183 return this->emitGetFnPtr(
Func, E);
4186template <
class Emitter>
4190 auto canonType = [](
const Type *T) {
4191 return T->getCanonicalTypeUnqualified().getTypePtr();
4199 return this->emitGetTypeid(
4203 return this->emitGetTypeid(
4212 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
4218 if (!this->emitGetTypeidPtr(TypeInfoType, E))
4221 return this->emitPopPtr(E);
4225template <
class Emitter>
4229 return this->emitDummyPtr(E, E);
4230 return this->emitError(E);
4233template <
class Emitter>
4236 return this->emitDummyPtr(E, E);
4237 return this->emitError(E);
4240template <
class Emitter>
4242 assert(
Ctx.getLangOpts().CPlusPlus);
4243 return this->emitConstBool(E->
getValue(), E);
4246template <
class Emitter>
4258 return this->emitDummyPtr(GuidDecl, E);
4263 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
4272 assert(
V.isStruct());
4273 assert(
V.getStructNumBases() == 0);
4277 return this->emitFinishInit(E);
4280template <
class Emitter>
4290template <
class Emitter>
4299template <
class Emitter>
4305template <
class Emitter>
4309 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
4313 if (OVE->isUnique())
4329template <
class Emitter>
4334template <
class Emitter>
4336 return this->emitError(E);
4339template <
class Emitter>
4345 return this->emitDummyPtr(E, E);
4348template <
class Emitter>
4352 QualType ElemType = VT->getElementType();
4356 PrimType SrcElemT = classifyVectorElementType(SrcType);
4358 unsigned SrcOffset =
4360 if (!this->
visit(Src))
4362 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4365 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4366 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4368 if (!this->emitArrayElemPop(SrcElemT, I, E))
4372 if (SrcElemT != ElemT) {
4373 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4375 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4376 const auto *TargetSemantics = &
Ctx.getFloatSemantics(ElemType);
4380 if (!this->emitInitElem(ElemT, I, E))
4387template <
class Emitter>
4391 return this->emitInvalid(E);
4400 assert(NumOutputElems > 0);
4406 if (!this->emitGetPtrLocal(*LocalIndex, E))
4411 unsigned VectorOffsets[2];
4412 for (
unsigned I = 0; I != 2; ++I) {
4415 if (!this->
visit(Vecs[I]))
4417 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4420 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4422 assert(ShuffleIndex >= -1);
4423 if (ShuffleIndex == -1)
4424 return this->emitInvalidShuffleVectorIndex(I, E);
4426 assert(ShuffleIndex < (NumInputElems * 2));
4427 if (!this->emitGetLocal(
PT_Ptr,
4428 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4430 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4431 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4434 if (!this->emitInitElem(ElemT, I, E))
4439 return this->emitPopPtr(E);
4444template <
class Emitter>
4449 Base->getType()->isVectorType() ||
4455 if (Indices.size() == 1) {
4460 if (!this->emitConstUint32(Indices[0], E))
4462 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4472 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4480 if (!this->emitGetPtrLocal(*ResultIndex, E))
4488 uint32_t DstIndex = 0;
4489 for (uint32_t I : Indices) {
4490 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4492 if (!this->emitArrayElemPop(ElemT, I, E))
4494 if (!this->emitInitElem(ElemT, DstIndex, E))
4504template <
class Emitter>
4508 return this->
discard(SubExpr) && this->emitInvalid(E);
4514 return this->emitDummyPtr(E, E);
4517template <
class Emitter>
4522 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4527 if (!this->
visit(SubExpr))
4529 if (!this->emitConstUint8(0, E))
4531 if (!this->emitArrayElemPtrPopUint8(E))
4533 if (!this->emitInitFieldPtr(R->getField(0u)->Offset, E))
4538 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4540 return this->emitInitField(SecondFieldT, R->getField(1u)->Offset, E);
4542 assert(SecondFieldT ==
PT_Ptr);
4544 if (!this->emitGetFieldPtr(R->getField(0u)->Offset, E))
4546 if (!this->emitExpandPtr(E))
4550 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4552 return this->emitInitFieldPtr(R->getField(1u)->Offset, E);
4555template <
class Emitter>
4570 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4572 return this->emitUnsupported(E);
4581 return this->
Visit(E);
4588 return this->
Visit(E);
4592 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4595 if (
const auto *CE = dyn_cast<CastExpr>(E);
4597 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4604 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4607 if (
const auto *CE = dyn_cast<CastExpr>(E);
4608 CE && (CE->getCastKind() == CK_DerivedToBase ||
4609 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4610 CE->getCastKind() == CK_NoOp))
4630 if (!this->emitGetPtrLocal(*LocalIndex, E))
4640 return this->
Visit(E);
4643template <
class Emitter>
4649 return this->
Visit(E) && this->emitFinishInit(E);
4652template <
class Emitter>
4658 return this->
Visit(E) && this->emitFinishInitPop(E);
4664 return this->
Visit(E);
4675 if (!this->
visit(E))
4677 return this->emitComplexBoolCast(E);
4682 if (!this->
visit(E))
4690 return this->emitIsNonNullPtr(E);
4694 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4697 return this->emitCast(*T,
PT_Bool, E);
4700template <
class Emitter>
4704 QT = AT->getValueType();
4708 return this->emitZeroBool(E);
4710 return this->emitZeroSint8(E);
4712 return this->emitZeroUint8(E);
4714 return this->emitZeroSint16(E);
4716 return this->emitZeroUint16(E);
4718 return this->emitZeroSint32(E);
4720 return this->emitZeroUint32(E);
4722 return this->emitZeroSint64(E);
4724 return this->emitZeroUint64(E);
4726 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4728 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4730 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4733 return this->emitNullMemberPtr(0,
nullptr, E);
4735 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4736 return this->emitFloat(F, E);
4739 auto Sem = Ctx.getASTContext().getFixedPointSemantics(E->
getType());
4743 llvm_unreachable(
"unknown primitive type");
4746template <
class Emitter>
4747bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4752 for (
const Record::Field &Field :
R->fields()) {
4753 if (Field.isUnnamedBitField())
4760 if (!this->visitZeroInitializer(T, QT, E))
4763 if (!this->emitInitFieldActivate(T, Field.Offset, E))
4767 if (!this->emitInitField(T, Field.Offset, E))
4772 if (!this->emitGetPtrField(Field.Offset, E))
4778 for (uint32_t I = 0, N = D->
getNumElems(); I != N; ++I) {
4779 if (!this->visitZeroInitializer(T, ET, E))
4781 if (!this->emitInitElem(T, I, E))
4786 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4789 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4797 if (!this->emitFinishInitActivatePop(E))
4801 if (!this->emitFinishInitPop(E))
4805 for (
const Record::Base &B :
R->bases()) {
4806 if (!this->emitGetPtrBase(B.Offset, E))
4808 if (!this->visitZeroRecordInitializer(B.R, E))
4810 if (!this->emitFinishInitPop(E))
4819template <
class Emitter>
4820bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4821 assert(T->isArrayType() || T->isAnyComplexType() || T->isVectorType());
4827 for (
size_t I = 0; I != NumElems; ++I) {
4828 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4830 if (!this->emitInitElem(*ElemT, I, E))
4836 const Record *
R = getRecord(ElemType);
4840 for (
size_t I = 0; I != NumElems; ++I) {
4841 if (!this->emitConstUint32(I, E))
4843 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4845 if (!this->visitZeroRecordInitializer(R, E))
4847 if (!this->emitPopPtr(E))
4853 for (
size_t I = 0; I != NumElems; ++I) {
4854 if (!this->emitConstUint32(I, E))
4856 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4858 if (!this->visitZeroArrayInitializer(ElemType, E))
4860 if (!this->emitPopPtr(E))
4869template <
class Emitter>
4870bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4872 if (!canClassify(E->
getType()))
4875 if (!this->visit(RHS))
4877 if (!this->visit(LHS))
4884 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4888 bool Activates = refersToUnion(LHS);
4891 if (!this->emitFlip(
PT_Ptr, RHT, E))
4894 if (DiscardResult) {
4895 if (BitField && Activates)
4896 return this->emitStoreBitFieldActivatePop(RHT, E);
4898 return this->emitStoreBitFieldPop(RHT, E);
4900 return this->emitStoreActivatePop(RHT, E);
4902 return this->emitStorePop(RHT, E);
4905 auto maybeLoad = [&](
bool Result) ->
bool {
4911 return this->emitLoadPop(RHT, E);
4915 if (BitField && Activates)
4916 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4918 return maybeLoad(this->emitStoreBitField(RHT, E));
4920 return maybeLoad(this->emitStoreActivate(RHT, E));
4922 return maybeLoad(this->emitStore(RHT, E));
4925template <
class Emitter>
4926template <
typename T>
4930 return this->emitConstSint8(
Value, E);
4932 return this->emitConstUint8(
Value, E);
4934 return this->emitConstSint16(
Value, E);
4936 return this->emitConstUint16(
Value, E);
4938 return this->emitConstSint32(
Value, E);
4940 return this->emitConstUint32(
Value, E);
4942 return this->emitConstSint64(
Value, E);
4944 return this->emitConstUint64(
Value, E);
4946 return this->emitConstBool(
Value, E);
4953 llvm_unreachable(
"Invalid integral type");
4956 llvm_unreachable(
"unknown primitive type");
4959template <
class Emitter>
4960template <
typename T>
4961bool Compiler<Emitter>::emitConst(T
Value,
const Expr *E) {
4962 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
4965template <
class Emitter>
4969 return this->emitConstIntAPS(
Value, E);
4971 return this->emitConstIntAP(
Value, E);
4973 if (
Value.isSigned())
4974 return this->emitConst(
Value.getSExtValue(), Ty, E);
4975 return this->emitConst(
Value.getZExtValue(), Ty, E);
4978template <
class Emitter>
4982 return this->emitConstIntAPS(
Value, E);
4984 return this->emitConstIntAP(
Value, E);
4987 return this->emitConst(
Value.getSExtValue(), Ty, E);
4988 return this->emitConst(
Value.getZExtValue(), Ty, E);
4991template <
class Emitter>
4992bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
4993 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
4996template <
class Emitter>
5009 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
5010 Locals.insert({VD, Local});
5011 VarScope->addForScopeKind(Local, SC);
5012 return Local.Offset;
5015template <
class Emitter>
5020 bool IsTemporary =
false;
5021 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
5024 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
5025 Init = VarD->getInit();
5027 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
5038 return std::nullopt;
5043 Locals.insert({Key, Local});
5044 VarScope->addForScopeKind(Local, SC);
5045 return Local.Offset;
5048template <
class Emitter>
5058 return std::nullopt;
5068 return Local.Offset;
5071template <
class Emitter>
5073 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
5074 return PT->getPointeeType()->getAsCanonical<RecordType>();
5080 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
5084template <
class Emitter>
5086 return P.getOrCreateRecord(RD);
5089template <
class Emitter>
5091 return Ctx.getOrCreateFunction(FD);
5094template <
class Emitter>
5098 auto maybeDestroyLocals = [&]() ->
bool {
5099 if (DestroyToplevelScope)
5100 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
5101 return this->emitCheckAllocations(E);
5108 return this->emitRetVoid(E) && maybeDestroyLocals();
5116 return this->emitRet(*T, E) && maybeDestroyLocals();
5124 if (!this->emitGetPtrLocal(*LocalOffset, E))
5132 return this->emitRetValue(E) && maybeDestroyLocals();
5135 return maybeDestroyLocals() &&
false;
5138template <
class Emitter>
5150 if (
auto GlobalIndex =
P.getGlobal(VD)) {
5151 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5165template <
class Emitter>
5167 bool ConstantContext) {
5170 if (!ConstantContext) {
5184 auto GlobalIndex =
P.getGlobal(VD);
5185 assert(GlobalIndex);
5187 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
5190 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
5194 auto Local =
Locals.find(VD);
5195 assert(Local !=
Locals.end());
5197 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
5200 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
5210 auto GlobalIndex =
P.getGlobal(VD);
5211 assert(GlobalIndex);
5212 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5221 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
5224template <
class Emitter>
5234 if (!this->isActive())
5239 if (
Init &&
Init->isValueDependent())
5243 auto checkDecl = [&]() ->
bool {
5245 return !NeedsOp || this->emitCheckDecl(VD, VD);
5252 if (!
P.getGlobal(*GlobalIndex)->isInitialized())
5255 if (
P.isGlobalInitialized(*GlobalIndex))
5259 }
else if ((GlobalIndex =
P.createGlobal(VD,
Init))) {
5273 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
5276 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
5282 return this->emitFinishInitGlobal(
Init);
5292 if (!
Init ||
Init->getType()->isVoidType())
5301 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
5305 return this->emitSetLocal(*VarT, Offset, VD);
5313 if (!this->emitGetPtrLocal(*Offset,
Init))
5321template <
class Emitter>
5326 return this->emitConst(Val.
getInt(), ValType, E);
5329 return this->emitFloat(F, E);
5334 if (!this->emitGetMemberPtr(MemberDecl, E))
5340 if (!this->emitCopyMemberPtrPath(PathEntry, IsDerived, E))
5346 return this->emitNullMemberPtr(0,
nullptr, E);
5351 return this->emitNull(ValType, 0,
nullptr, E);
5356 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
5357 return this->
visit(BaseExpr);
5362 QualType EntryType = VD->getType();
5363 for (
auto &Entry : Path) {
5365 uint64_t Index = Entry.getAsArrayIndex();
5368 if (!this->emitConst(Index,
PT_Uint64, E))
5370 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
5372 EntryType = ElemType;
5382 const Decl *BaseOrMember = Entry.getAsBaseOrMember().getPointer();
5383 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
5385 if (!this->emitGetPtrFieldPop(EntryOffset, E))
5387 EntryType = FD->getType();
5391 if (!this->emitGetPtrBasePop(BaseOffset,
false, E))
5393 EntryType =
Ctx.getASTContext().getCanonicalTagType(
Base);
5405template <
class Emitter>
5413 const Record::Field *RF = R->getField(I);
5414 QualType FieldType = RF->Decl->getType();
5420 if (!this->emitInitField(*PT, RF->Offset, E))
5423 if (!this->emitGetPtrField(RF->Offset, E))
5427 if (!this->emitFinishInitPop(E))
5435 const Record::Base *RB = R->getBase(I);
5436 QualType BaseType =
Ctx.getASTContext().getCanonicalTagType(RB->Decl);
5438 if (!this->emitGetPtrBase(RB->Offset, E))
5442 if (!this->emitFinishInitPop(E))
5455 const Record::Field *RF = R->getField(UnionField);
5456 QualType FieldType = RF->Decl->getType();
5461 if (RF->isBitField())
5462 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
5464 return this->emitInitFieldActivate(*PT, RF->Offset, E);
5467 if (!this->emitGetPtrField(RF->Offset, E))
5469 if (!this->emitActivate(E))
5473 return this->emitPopPtr(E);
5476 const auto *ArrType = T->getAsArrayTypeUnsafe();
5477 QualType ElemType = ArrType->getElementType();
5478 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5483 if (!this->emitInitElem(*ElemT, A, E))
5486 if (!this->emitConstUint32(A, E))
5488 if (!this->emitArrayElemPtrUint32(E))
5492 if (!this->emitPopPtr(E))
5503template <
class Emitter>
5505 unsigned BuiltinID) {
5506 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5511 return this->emitConst(0, E);
5514 if (!this->emitStartSpeculation(E))
5516 LabelTy EndLabel = this->getLabel();
5517 if (!this->speculate(E, EndLabel))
5519 if (!this->emitEndSpeculation(E))
5521 this->fallthrough(EndLabel);
5529 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5530 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5531 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5532 BuiltinID == Builtin::BI__builtin_function_start) {
5535 return this->emitDummyPtr(E, E);
5546 if (!this->emitGetPtrLocal(*LocalIndex, E))
5551 switch (BuiltinID) {
5552 case Builtin::BI__builtin_object_size:
5553 case Builtin::BI__builtin_dynamic_object_size: {
5557 if (!this->
visit(Arg0))
5568 case Builtin::BI__assume:
5569 case Builtin::BI__builtin_assume:
5572 case Builtin::BI__atomic_is_lock_free:
5573 case Builtin::BI__atomic_always_lock_free: {
5584 for (
const auto *Arg : E->
arguments()) {
5585 if (!this->
visit(Arg))
5591 if (!this->emitCallBI(E, BuiltinID, E))
5600template <
class Emitter>
5619 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5620 DD && DD->isTrivial()) {
5622 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5624 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5625 this->emitPopPtr(E);
5633 bool HasRVO = !
ReturnType->isVoidType() && !T;
5641 if (!this->emitGetPtrLocal(*LocalIndex, E))
5649 if (!this->emitGetPtrLocal(*LocalIndex, E))
5653 if (!this->emitDupPtr(E))
5660 bool IsAssignmentOperatorCall =
false;
5661 bool ActivateLHS =
false;
5662 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5663 OCE && OCE->isAssignmentOp()) {
5667 assert(Args.size() == 2);
5668 const CXXRecordDecl *LHSRecord = Args[0]->getType()->getAsCXXRecordDecl();
5670 IsAssignmentOperatorCall =
true;
5671 std::reverse(Args.begin(), Args.end());
5677 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5678 MD && MD->isStatic()) {
5682 Args.erase(Args.begin());
5686 bool Devirtualized =
false;
5689 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5697 if (!this->
visit(Callee))
5699 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5701 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5703 if (!this->emitGetMemberPtrBase(E))
5706 const auto *InstancePtr = MC->getImplicitObjectArgument();
5713 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5714 Devirtualized =
true;
5715 if (!this->
visit(Stripped))
5718 if (!this->
visit(InstancePtr))
5722 if (!this->
visit(InstancePtr))
5726 }
else if (
const auto *PD =
5727 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5728 if (!this->emitCheckPseudoDtor(E))
5736 return this->emitEndLifetimePop(E);
5737 }
else if (!FuncDecl) {
5741 if (!this->
visit(Callee))
5743 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5752 if (IsAssignmentOperatorCall) {
5753 assert(Args.size() == 2);
5756 if (!this->emitFlip(Arg2T, Arg1T, E))
5770 assert(HasRVO ==
Func->hasRVO());
5772 bool HasQualifier =
false;
5773 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5774 HasQualifier = ME->hasQualifier();
5776 bool IsVirtual =
false;
5777 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5778 IsVirtual = !Devirtualized && MD->isVirtual();
5783 if (IsVirtual && !HasQualifier) {
5784 uint32_t VarArgSize = 0;
5785 unsigned NumParams =
5787 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5790 if (!this->emitCallVirt(
Func, VarArgSize, E))
5792 }
else if (
Func->isVariadic()) {
5793 uint32_t VarArgSize = 0;
5794 unsigned NumParams =
5796 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5798 if (!this->emitCallVar(
Func, VarArgSize, E))
5801 if (!this->emitCall(
Func, 0, E))
5810 uint32_t ArgSize = 0;
5811 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5817 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5819 if (!this->emitGetMemberPtrDecl(E))
5822 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5825 if (!this->emitCallPtr(ArgSize, E, E))
5836template <
class Emitter>
5843template <
class Emitter>
5850template <
class Emitter>
5855 return this->emitConstBool(E->
getValue(), E);
5858template <
class Emitter>
5864 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5865 return this->emitNullPtr(Val,
nullptr, E);
5868template <
class Emitter>
5876 return this->emitZero(T, E);
5879template <
class Emitter>
5884 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
5885 if (this->LambdaThisCapture.Offset > 0) {
5886 if (this->LambdaThisCapture.IsPtr)
5887 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
5888 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
5897 return this->emitThis(E);
5912 unsigned StartIndex = 0;
5913 unsigned EndIndex = 0;
5915 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
5917 EndIndex = StartIndex;
5924 for (; StartIndex > 0; --StartIndex) {
5934 if (StartIndex == 0 && EndIndex == 0)
5937 assert(StartIndex < EndIndex);
5940 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
5952 case Stmt::CompoundStmtClass:
5954 case Stmt::DeclStmtClass:
5956 case Stmt::ReturnStmtClass:
5958 case Stmt::IfStmtClass:
5960 case Stmt::WhileStmtClass:
5962 case Stmt::DoStmtClass:
5964 case Stmt::ForStmtClass:
5966 case Stmt::CXXForRangeStmtClass:
5968 case Stmt::BreakStmtClass:
5970 case Stmt::ContinueStmtClass:
5972 case Stmt::SwitchStmtClass:
5974 case Stmt::CaseStmtClass:
5976 case Stmt::DefaultStmtClass:
5978 case Stmt::AttributedStmtClass:
5980 case Stmt::CXXTryStmtClass:
5982 case Stmt::NullStmtClass:
5985 case Stmt::GCCAsmStmtClass:
5986 case Stmt::MSAsmStmtClass:
5987 case Stmt::GotoStmtClass:
5988 return this->emitInvalid(S);
5989 case Stmt::LabelStmtClass:
5992 if (
const auto *E = dyn_cast<Expr>(S))
5999template <
class Emitter>
6002 for (
const auto *InnerStmt : S->
body())
6005 return Scope.destroyLocals();
6008template <
class Emitter>
6009bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
6010 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
6011 for (
auto *BD : DD->flat_bindings())
6012 if (
auto *KD = BD->getHoldingVar();
6013 KD && !this->visitVarDecl(KD, KD->getInit()))
6027template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
6029 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
6030 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
6037 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
6042 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
6043 ICE && (ICE->getCastKind() == CK_NoOp ||
6044 ICE->getCastKind() == CK_DerivedToBase ||
6045 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
6046 E = ICE->getSubExpr();
6050 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
6051 const auto *ThisRecord =
6052 This->getType()->getPointeeType()->getAsRecordDecl();
6053 if (!ThisRecord->isUnion())
6056 if (
const auto *Ctor =
6057 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
6058 return Ctor->getParent() == ThisRecord;
6067template <
class Emitter>
6069 bool EvaluateConditionDecl) {
6070 for (
const auto *D : DS->
decls()) {
6075 const auto *VD = dyn_cast<VarDecl>(D);
6082 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
6089template <
class Emitter>
6092 return this->emitUnsupported(RS);
6098 if (!this->
visit(RE))
6104 if (RE->getType()->isVoidType()) {
6105 if (!this->
visit(RE))
6108 if (RE->containsErrors())
6113 if (!this->emitRVOPtr(RE))
6119 return this->emitRetVoid(RS);
6125 return this->emitRetVoid(RS);
6131 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
6138 if (
auto *CondInit = IS->
getInit()) {
6154 return visitChildStmt(IS->
getThen());
6156 return visitChildStmt(Else);
6162 if (!this->emitIsConstantContext(IS))
6165 if (!this->emitIsConstantContext(IS))
6167 if (!this->emitInv(IS))
6181 LabelTy LabelElse = this->getLabel();
6182 LabelTy LabelEnd = this->getLabel();
6183 if (!this->jumpFalse(LabelElse, IS))
6185 if (!visitChildStmt(IS->
getThen()))
6187 if (!this->jump(LabelEnd, IS))
6189 this->emitLabel(LabelElse);
6190 if (!visitChildStmt(Else))
6192 this->emitLabel(LabelEnd);
6194 LabelTy LabelEnd = this->getLabel();
6195 if (!this->jumpFalse(LabelEnd, IS))
6197 if (!visitChildStmt(IS->
getThen()))
6199 this->emitLabel(LabelEnd);
6208template <
class Emitter>
6213 LabelTy CondLabel = this->getLabel();
6214 LabelTy EndLabel = this->getLabel();
6218 this->fallthrough(CondLabel);
6219 this->emitLabel(CondLabel);
6235 if (!this->jumpFalse(EndLabel, S))
6245 if (!this->jump(CondLabel, S))
6247 this->fallthrough(EndLabel);
6248 this->emitLabel(EndLabel);
6257 LabelTy StartLabel = this->getLabel();
6258 LabelTy EndLabel = this->getLabel();
6259 LabelTy CondLabel = this->getLabel();
6263 this->fallthrough(StartLabel);
6264 this->emitLabel(StartLabel);
6270 this->fallthrough(CondLabel);
6271 this->emitLabel(CondLabel);
6278 if (!this->jumpTrue(StartLabel, S))
6281 this->fallthrough(EndLabel);
6282 this->emitLabel(EndLabel);
6286template <
class Emitter>
6294 LabelTy EndLabel = this->getLabel();
6295 LabelTy CondLabel = this->getLabel();
6296 LabelTy IncLabel = this->getLabel();
6303 this->fallthrough(CondLabel);
6304 this->emitLabel(CondLabel);
6316 if (!this->jumpFalse(EndLabel, S))
6325 this->fallthrough(IncLabel);
6326 this->emitLabel(IncLabel);
6332 if (!this->jump(CondLabel, S))
6336 this->emitLabel(EndLabel);
6342template <
class Emitter>
6352 LabelTy EndLabel = this->getLabel();
6353 LabelTy CondLabel = this->getLabel();
6354 LabelTy IncLabel = this->getLabel();
6369 this->fallthrough(CondLabel);
6370 this->emitLabel(CondLabel);
6373 if (!this->jumpFalse(EndLabel, S))
6384 this->fallthrough(IncLabel);
6385 this->emitLabel(IncLabel);
6390 if (!this->jump(CondLabel, S))
6393 this->fallthrough(EndLabel);
6394 this->emitLabel(EndLabel);
6398template <
class Emitter>
6409 if (LI.BreakLabel) {
6410 TargetLabel = *LI.BreakLabel;
6411 BreakScope = LI.BreakOrContinueScope;
6417 if (LI.Name == TargetLoop) {
6418 TargetLabel = *LI.BreakLabel;
6419 BreakScope = LI.BreakOrContinueScope;
6430 C =
C->getParent()) {
6431 if (!
C->destroyLocals())
6435 return this->jump(*TargetLabel, S);
6438template <
class Emitter>
6449 if (LI.ContinueLabel) {
6450 TargetLabel = *LI.ContinueLabel;
6451 ContinueScope = LI.BreakOrContinueScope;
6457 if (LI.Name == TargetLoop) {
6458 TargetLabel = *LI.ContinueLabel;
6459 ContinueScope = LI.BreakOrContinueScope;
6464 assert(TargetLabel);
6467 C =
C->getParent()) {
6468 if (!
C->destroyLocals())
6472 return this->jump(*TargetLabel, S);
6475template <
class Emitter>
6478 if (
Cond->containsErrors())
6484 LabelTy EndLabel = this->getLabel();
6489 if (
const auto *CondInit = S->
getInit())
6500 if (!this->emitSetLocal(CondT, CondVar, S))
6510 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6513 if (CS->caseStmtIsGNURange()) {
6514 LabelTy EndOfRangeCheck = this->getLabel();
6515 const Expr *Low = CS->getLHS();
6516 const Expr *High = CS->getRHS();
6520 if (!this->emitGetLocal(CondT, CondVar, CS))
6522 if (!this->
visit(Low))
6525 if (!this->emitGE(
LT, S))
6527 if (!this->jumpFalse(EndOfRangeCheck, S))
6530 if (!this->emitGetLocal(CondT, CondVar, CS))
6532 if (!this->
visit(High))
6535 if (!this->emitLE(HT, S))
6539 this->emitLabel(EndOfRangeCheck);
6544 if (
Value->isValueDependent())
6549 if (!this->emitGetLocal(CondT, CondVar, CS))
6555 if (!this->emitEQ(ValueT, S))
6560 assert(!DefaultLabel);
6561 DefaultLabel = this->getLabel();
6568 if (!this->jump(*DefaultLabel, S))
6571 if (!this->jump(EndLabel, S))
6579 this->fallthrough(EndLabel);
6580 this->emitLabel(EndLabel);
6585template <
class Emitter>
6592template <
class Emitter>
6599 if (LI.DefaultLabel) {
6600 DefaultLabel = *LI.DefaultLabel;
6605 this->emitLabel(DefaultLabel);
6609template <
class Emitter>
6616 if (IsMSVCConstexprAttr && !this->emitPushMSVCCE(S))
6619 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6620 !this->Ctx.getLangOpts().MSVCCompat) {
6622 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6628 const Expr *Assumption = AA->getAssumption();
6639 if (!this->emitAssume(Assumption))
6648 if (IsMSVCConstexprAttr)
6649 return this->emitPopMSVCCE(S);
6653template <
class Emitter>
6659template <
class Emitter>
6660bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6667 assert(ClosureClass->
captures().empty());
6671 "A generic lambda's static-invoker function must be a "
6672 "template specialization");
6676 void *InsertPos =
nullptr;
6677 const FunctionDecl *CorrespondingCallOpSpecialization =
6679 assert(CorrespondingCallOpSpecialization);
6680 LambdaCallOp = CorrespondingCallOpSpecialization;
6684 assert(ClosureClass->
captures().empty());
6685 const Function *
Func = this->getFunction(LambdaCallOp);
6688 assert(
Func->hasThisPointer());
6691 if (
Func->hasRVO()) {
6692 if (!this->emitRVOPtr(MD))
6700 if (!this->emitNullPtr(0,
nullptr, MD))
6705 auto It = this->Params.find(PVD);
6706 assert(It != this->Params.end());
6710 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6711 if (!this->emitGetParam(ParamType, It->second.Index, MD))
6715 if (!this->emitCall(
Func, 0, LambdaCallOp))
6720 return this->emitRet(*ReturnType, MD);
6723 return this->emitRetVoid(MD);
6726template <
class Emitter>
6727bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6728 if (Ctx.getLangOpts().CPlusPlus23)
6738 const Expr *InitExpr =
Init->getInit();
6740 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6744 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6754template <
class Emitter>
6756 assert(!ReturnType);
6759 if (!this->emitStartThisLifetime1(Ctor))
6762 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6763 const Expr *InitExpr,
6766 if (InitExpr->getType().isNull())
6770 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6773 if (!this->visit(InitExpr))
6776 if (F->isBitField())
6777 return this->emitInitThisBitField(*T, FieldOffset, F->bitWidth(),
6779 return this->emitInitThisField(*T, FieldOffset, InitExpr);
6784 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6787 if (
Activate && !this->emitActivate(InitExpr))
6790 return this->visitInitializerPop(InitExpr);
6794 const Record *
R = this->getRecord(RD);
6797 bool IsUnion =
R->isUnion();
6802 if (
R->getNumFields() == 0)
6803 return this->emitRetVoid(Ctor);
6806 if (!this->emitThis(Ctor))
6809 if (!this->emitGetParam(
PT_Ptr, 0, Ctor))
6812 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6813 this->emitRetVoid(Ctor);
6816 unsigned FieldInits = 0;
6818 for (
const auto *
Init : Ctor->
inits()) {
6822 const Expr *InitExpr =
Init->getInit();
6824 const Record::Field *F =
R->getField(
Member);
6828 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6832 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6835 if (
Init->isBaseVirtual()) {
6836 assert(
R->getVirtualBase(BaseDecl));
6837 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6843 const Record::Base *B =
R->getBase(BaseDecl);
6845 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6849 if (IsUnion && !this->emitActivate(InitExpr))
6852 if (!this->visitInitializerPop(InitExpr))
6857 unsigned ChainSize = IFD->getChainingSize();
6858 assert(ChainSize >= 2);
6860 unsigned NestedFieldOffset = 0;
6861 const Record::Field *NestedField =
nullptr;
6862 for (
unsigned I = 0; I != ChainSize; ++I) {
6864 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6865 assert(FieldRecord);
6867 NestedField = FieldRecord->
getField(FD);
6868 assert(NestedField);
6869 IsUnion = IsUnion || FieldRecord->
isUnion();
6871 NestedFieldOffset += NestedField->Offset;
6874 if (I != ChainSize - 1)
6877 assert(NestedField);
6880 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
6885 unsigned InitFieldOffset = 0;
6886 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
6888 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6889 assert(FieldRecord);
6890 NestedField = FieldRecord->
getField(FD);
6891 InitFieldOffset += NestedField->Offset;
6892 assert(NestedField);
6893 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
6895 if (!this->emitFinishInitPop(InitExpr))
6899 InitStack.pop_back_n(ChainSize - 1);
6902 assert(
Init->isDelegatingInitializer());
6903 if (!this->emitThis(InitExpr))
6905 if (!this->visitInitializerPop(
Init->getInit()))
6909 if (!
Scope.destroyLocals())
6913 if (FieldInits !=
R->getNumFields()) {
6914 assert(FieldInits < R->getNumFields());
6916 if (!this->emitStartThisLifetime(Ctor))
6923 if (
const auto *CS = dyn_cast<CompoundStmt>(Body)) {
6924 if (!CS->body_empty() && !this->emitCtorCheck(
SourceInfo{}))
6931 if (!visitStmt(Body))
6938template <
class Emitter>
6941 const Record *
R = this->getRecord(RD);
6946 if (!this->visitStmt(Dtor->
getBody()))
6950 if (!this->emitThis(Dtor))
6953 if (!this->emitCheckDestruction(Dtor))
6957 if (!
R->isUnion()) {
6961 for (
const Record::Field &Field : llvm::reverse(
R->fields())) {
6965 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
6967 if (!this->emitDestructionPop(D,
SourceInfo{}))
6972 for (
const Record::Base &
Base : llvm::reverse(
R->bases())) {
6973 if (
Base.R->hasTrivialDtor())
6977 if (!this->emitRecordDestructionPop(
Base.R, {}))
6981 if (!this->emitMarkDestroyed(Dtor))
6985 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
6988template <
class Emitter>
6989bool Compiler<Emitter>::compileUnionAssignmentOperator(
6991 if (!this->emitThis(MD))
6994 if (!this->emitGetParam(
PT_Ptr, 0, MD))
6997 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
7000template <
class Emitter>
7010 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
7011 return this->compileConstructor(Ctor);
7012 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
7013 return this->compileDestructor(Dtor);
7016 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
7021 return this->compileUnionAssignmentOperator(MD);
7024 return this->emitLambdaStaticInvokerBody(MD);
7028 if (
const auto *Body = F->
getBody())
7042 return FD->getBitWidthValue();
7045template <
class Emitter>
7058 if (!
Ctx.getLangOpts().CPlusPlus14)
7059 return this->emitInvalid(E);
7061 return this->emitError(E);
7063 if (!this->
visit(SubExpr))
7067 if (!this->emitIncPtr(E))
7074 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
7075 : this->emitIncf(getFPOptions(E), E);
7087 if (!
Ctx.getLangOpts().CPlusPlus14)
7088 return this->emitInvalid(E);
7090 return this->emitError(E);
7092 if (!this->
visit(SubExpr))
7096 if (!this->emitDecPtr(E))
7103 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
7104 : this->emitDecf(getFPOptions(E), E);
7117 if (!
Ctx.getLangOpts().CPlusPlus14)
7118 return this->emitInvalid(E);
7120 return this->emitError(E);
7122 if (!this->
visit(SubExpr))
7126 if (!this->emitLoadPtr(E))
7128 if (!this->emitConstUint8(1, E))
7130 if (!this->emitAddOffsetUint8(E))
7132 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7138 return this->emitIncfPop(getFPOptions(E), E);
7148 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7149 if (!this->emitLoadFloat(E))
7151 APFloat F(TargetSemantics, 1);
7152 if (!this->emitFloat(F, E))
7155 if (!this->emitAddf(getFPOptions(E), E))
7157 if (!this->emitStoreFloat(E))
7169 return E->
isGLValue() || this->emitLoadPop(*T, E);
7172 if (!
Ctx.getLangOpts().CPlusPlus14)
7173 return this->emitInvalid(E);
7175 return this->emitError(E);
7177 if (!this->
visit(SubExpr))
7181 if (!this->emitLoadPtr(E))
7183 if (!this->emitConstUint8(1, E))
7185 if (!this->emitSubOffsetUint8(E))
7187 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7193 return this->emitDecfPop(getFPOptions(E), E);
7203 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7204 if (!this->emitLoadFloat(E))
7206 APFloat F(TargetSemantics, 1);
7207 if (!this->emitFloat(F, E))
7210 if (!this->emitSubf(getFPOptions(E), E))
7212 if (!this->emitStoreFloat(E))
7224 return E->
isGLValue() || this->emitLoadPop(*T, E);
7228 return this->emitError(E);
7231 return this->
discard(SubExpr);
7236 if (!this->emitInv(E))
7240 return this->emitCast(
PT_Bool, ET, E);
7244 return this->emitError(E);
7246 if (!this->
visit(SubExpr))
7248 return DiscardResult ? this->emitPop(*T, E) : this->emitNeg(*T, E);
7251 return this->emitError(E);
7253 if (!this->
visit(SubExpr))
7268 return this->
discard(SubExpr);
7270 if (!this->
visit(SubExpr))
7277 return this->emitNarrowPtr(E);
7282 return this->emitError(E);
7284 if (!this->
visit(SubExpr))
7286 return DiscardResult ? this->emitPop(*T, E) : this->emitComp(*T, E);
7298 : this->visitZeroInitializer(*T, SubExpr->
getType(), SubExpr);
7303 assert(
false &&
"Unhandled opcode");
7309template <
class Emitter>
7315 return this->
discard(SubExpr);
7318 auto prepareResult = [=]() ->
bool {
7323 return this->emitGetPtrLocal(*LocalIndex, E);
7330 unsigned SubExprOffset = ~0u;
7331 auto createTemp = [=, &SubExprOffset]() ->
bool {
7334 if (!this->
visit(SubExpr))
7336 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
7340 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7341 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7343 return this->emitArrayElemPop(ElemT, Index, E);
7348 if (!prepareResult())
7352 for (
unsigned I = 0; I != 2; ++I) {
7353 if (!getElem(SubExprOffset, I))
7355 if (!this->emitNeg(ElemT, E))
7357 if (!this->emitInitElem(ElemT, I, E))
7368 if (!this->
visit(SubExpr))
7370 if (!this->emitComplexBoolCast(SubExpr))
7372 if (!this->emitInv(E))
7375 return this->emitCast(
PT_Bool, ET, E);
7379 return this->emitComplexReal(SubExpr);
7382 if (!this->
visit(SubExpr))
7386 if (!this->emitConstUint8(1, E))
7388 return this->emitArrayElemPtrPopUint8(E);
7399 if (!this->emitArrayElem(ElemT, 1, E))
7401 if (!this->emitNeg(ElemT, E))
7403 if (!this->emitInitElem(ElemT, 1, E))
7411 return this->emitInvalid(E);
7417template <
class Emitter>
7423 return this->
discard(SubExpr);
7426 if (UnaryOp == UO_Extension)
7429 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
7430 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
7431 return this->emitInvalid(E);
7434 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
7441 if (!this->emitGetPtrLocal(*LocalIndex, E))
7446 unsigned SubExprOffset =
7448 if (!this->
visit(SubExpr))
7450 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
7455 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7456 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7458 return this->emitArrayElemPop(ElemT, Index, E);
7463 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7464 if (!getElem(SubExprOffset, I))
7466 if (!this->emitNeg(ElemT, E))
7468 if (!this->emitInitElem(ElemT, I, E))
7483 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7484 if (!getElem(SubExprOffset, I))
7487 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
7489 if (!this->emitInv(E))
7491 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
7493 if (!this->emitNeg(ElemT, E))
7495 if (ElemT != ResultVecElemT &&
7496 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
7498 if (!this->emitInitElem(ResultVecElemT, I, E))
7504 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7505 if (!getElem(SubExprOffset, I))
7508 if (!this->emitInv(E))
7511 if (!this->emitComp(ElemT, E))
7514 if (!this->emitInitElem(ElemT, I, E))
7519 llvm_unreachable(
"Unsupported unary operators should be handled up front");
7524template <
class Emitter>
7529 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D))
7530 return this->emitConst(ECD->getInitVal(), E);
7531 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
7533 return F && this->emitGetFnPtr(F, E);
7535 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
7540 return this->emitInitGlobal(*T, *Index, E);
7543 if (!this->emitGetPtrGlobal(*Index, E))
7547 return this->emitFinishInit(E);
7562 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
7563 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7568 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7569 if (IsReference || !It->second.IsPtr)
7570 return this->emitGetParam(
classifyPrim(E), It->second.Index, E);
7572 return this->emitGetPtrParam(It->second.Index, E);
7575 if (!
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7581 const unsigned Offset = It->second.Offset;
7584 return this->emitGetRefLocal(Offset, E);
7586 return this->emitGetPtrLocal(Offset, E);
7589 if (
auto GlobalIndex =
P.getGlobal(D)) {
7591 if (!
Ctx.getLangOpts().CPlusPlus11)
7592 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7593 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7596 return this->emitGetPtrGlobal(*GlobalIndex, E);
7600 auto revisit = [&](
const VarDecl *VD,
7601 bool IsConstexprUnknown =
true) ->
bool {
7603 IsConstexprUnknown);
7608 if (!this->emitPopCC(E))
7611 if (VarState.notCreated())
7619 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
7621 if (
auto It = this->LambdaCaptures.find(D);
7622 It != this->LambdaCaptures.end()) {
7623 auto [Offset, IsPtr] = It->second;
7626 return this->emitGetThisFieldPtr(Offset, E);
7627 return this->emitGetPtrThisField(Offset, E);
7631 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7632 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7633 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7637 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7638 return this->
visit(BD->getBinding());
7642 return this->emitDummyPtr(D, E);
7646 const auto *VD = dyn_cast<VarDecl>(D);
7648 return this->emitError(E);
7651 if (!
Ctx.getLangOpts().CPlusPlus) {
7654 return revisit(VD,
false);
7655 return this->emitDummyPtr(D, E);
7659 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7660 if (T.isConstant(
Ctx.getASTContext()))
7662 return T->isReferenceType();
7666 typeShouldBeVisited(DeclType)) {
7668 Init && !
Init->isValueDependent()) {
7674 (void)
Init->EvaluateAsInitializer(
V,
Ctx.getASTContext(), VD, Notes,
7688 bool IsConstexprUnknown = !DeclType.
isConstant(
Ctx.getASTContext()) &&
7693 return revisit(VD, IsConstexprUnknown);
7694 }
else if (
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7695 return revisit(VD,
true);
7702 return this->emitDummyPtr(D, E);
7705template <
class Emitter>
7713 if (!
C->destroyLocals())
7719template <
class Emitter>
7720unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7723 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7725 return Ty->getAsCXXRecordDecl();
7727 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7728 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7730 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7734template <
class Emitter>
7741 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7746 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7747 getFPOptions(E), E);
7749 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7750 getFPOptions(E), E);
7754 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7759 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7761 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7765 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7769 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7770 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7777template <
class Emitter>
7780 assert(FromT != ToT);
7783 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7785 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7787 return this->emitCast(FromT, ToT, E);
7791template <
class Emitter>
7792bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7796 return this->
discard(SubExpr);
7798 if (!this->visit(SubExpr))
7801 if (!this->emitConstUint8(0, SubExpr))
7803 return this->emitArrayElemPtrPopUint8(SubExpr);
7807 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7811template <
class Emitter>
7812bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7813 assert(!DiscardResult);
7817 if (!this->emitArrayElem(ElemT, 0, E))
7820 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7823 if (!this->emitCast(ElemT,
PT_Bool, E))
7828 LabelTy LabelTrue = this->getLabel();
7829 if (!this->jumpTrue(LabelTrue, E))
7832 if (!this->emitArrayElemPop(ElemT, 1, E))
7835 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7838 if (!this->emitCast(ElemT,
PT_Bool, E))
7842 LabelTy EndLabel = this->getLabel();
7843 this->jump(EndLabel, E);
7845 this->emitLabel(LabelTrue);
7846 if (!this->emitPopPtr(E))
7848 if (!this->emitConstBool(
true, E))
7851 this->fallthrough(EndLabel);
7852 this->emitLabel(EndLabel);
7857template <
class Emitter>
7858bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
7869 LHSIsComplex =
true;
7870 ElemT = classifyComplexElementType(LHS->
getType());
7871 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
7872 if (!this->visit(LHS))
7874 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
7877 LHSIsComplex =
false;
7879 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
7880 if (!this->visit(LHS))
7882 if (!this->emitSetLocal(LHST, LHSOffset, E))
7889 RHSIsComplex =
true;
7890 ElemT = classifyComplexElementType(RHS->
getType());
7891 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
7892 if (!this->visit(RHS))
7894 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
7897 RHSIsComplex =
false;
7899 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
7900 if (!this->visit(RHS))
7902 if (!this->emitSetLocal(RHST, RHSOffset, E))
7906 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
7907 bool IsComplex) ->
bool {
7909 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
7911 return this->emitArrayElemPop(ElemT, Index, E);
7913 return this->emitGetLocal(ElemT, LocalOffset, E);
7916 for (
unsigned I = 0; I != 2; ++I) {
7918 if (!getElem(LHSOffset, I, LHSIsComplex))
7920 if (!getElem(RHSOffset, I, RHSIsComplex))
7923 if (!this->emitEQ(ElemT, E))
7926 if (!this->emitCastBoolUint8(E))
7931 if (!this->emitAddUint8(E))
7933 if (!this->emitConstUint8(2, E))
7937 if (!this->emitEQUint8(E))
7940 if (!this->emitNEUint8(E))
7947 return this->emitCast(
PT_Bool, ResT, E);
7954template <
class Emitter>
7955bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
7958 assert(!
R->hasTrivialDtor());
7961 const Function *DtorFunc = getFunction(Dtor);
7964 assert(DtorFunc->hasThisPointer());
7965 assert(DtorFunc->getNumParams() == 1);
7966 return this->emitCall(DtorFunc, 0, Loc);
7971template <
class Emitter>
7972bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
7984 return this->emitPopPtr(Loc);
7986 for (ssize_t I = N - 1; I >= 1; --I) {
7987 if (!this->emitConstUint64(I, Loc))
7989 if (!this->emitArrayElemPtrUint64(Loc))
7991 if (!this->emitDestructionPop(ElemDesc, Loc))
7995 if (!this->emitConstUint64(0, Loc))
7997 if (!this->emitArrayElemPtrPopUint64(Loc))
7999 return this->emitDestructionPop(ElemDesc, Loc);
8004 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
8009template <
class Emitter>
8010bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E) {
8011 assert(!DiscardResult &&
"Should've been checked before");
8012 unsigned DummyID = P.getOrCreateDummy(D);
8014 if (!this->emitGetPtrGlobal(DummyID, E))
8022 return this->emitDecayPtr(
PT_Ptr, PT, E);
8028template <
class Emitter>
8029bool Compiler<Emitter>::emitFloat(
const APFloat &F,
const Expr *E) {
8031 return this->emitConstFloat(
Floating(F), E);
8033 APInt I = F.bitcastToAPInt();
8034 return this->emitConstFloat(
8035 Floating(
const_cast<uint64_t *
>(I.getRawData()),
8036 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
8047template <
class Emitter>
8048bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
8061 if (!this->emitGetPtrLocal(*LocalIndex, E))
8071 if (!this->visit(SubExpr))
8073 }
else if (
OptPrimType FromT = classify(SubExpr)) {
8074 unsigned TempOffset =
8075 allocateLocalPrimitive(SubExpr, *FromT,
true);
8076 if (!this->visit(SubExpr))
8078 if (!this->emitSetLocal(*FromT, TempOffset, E))
8080 if (!this->emitGetPtrLocal(TempOffset, E))
8087 if (!this->emitBitCast(E))
8089 return DiscardResult ? this->emitPopPtr(E) :
true;
8093 const llvm::fltSemantics *TargetSemantics =
nullptr;
8095 TargetSemantics = &Ctx.getFloatSemantics(ToType);
8101 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
8103 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
8104 ResultBitWidth, TargetSemantics,
8109 return this->emitPop(*ToT, E);
8118template <
class Emitter>
8119bool Compiler<Emitter>::emitHLSLAggregateSplat(
PrimType SrcT,
8124 unsigned NumElems = 0;
8127 NumElems = VT->getNumElements();
8128 ElemType = VT->getElementType();
8130 NumElems = MT->getNumElementsFlattened();
8131 ElemType = MT->getElementType();
8134 PrimType ElemT = classifyPrim(ElemType);
8135 for (
unsigned I = 0; I != NumElems; ++I) {
8136 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8138 if (!this->emitPrimCast(SrcT, ElemT, ElemType, E))
8140 if (!this->emitInitElem(ElemT, I, E))
8150 QualType ArrElemType = CAT->getElementType();
8151 unsigned ArrSize = CAT->getZExtSize();
8154 for (
unsigned I = 0; I != ArrSize; ++I) {
8155 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8157 if (!this->emitPrimCast(SrcT, *ElemT, ArrElemType, E))
8159 if (!this->emitInitElem(*ElemT, I, E))
8163 for (
unsigned I = 0; I != ArrSize; ++I) {
8164 if (!this->emitConstUint32(I, E))
8166 if (!this->emitArrayElemPtrUint32(E))
8168 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, ArrElemType, E))
8170 if (!this->emitFinishInitPop(E))
8180 const Record *
R = getRecord(DestType);
8184 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8186 const Record::Base *B =
R->getBase(BS.getType());
8188 if (!this->emitGetPtrBase(B->Offset, E))
8190 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, BS.getType(), E))
8192 if (!this->emitFinishInitPop(E))
8197 for (
const Record::Field &F :
R->fields()) {
8198 if (F.isUnnamedBitField())
8201 QualType FieldType = F.Decl->getType();
8203 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8205 if (!this->emitPrimCast(SrcT, *FieldT, FieldType, E))
8207 if (F.isBitField()) {
8208 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8211 if (!this->emitInitField(*FieldT, F.Offset, E))
8215 if (!this->emitGetPtrField(F.Offset, E))
8217 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, FieldType, E))
8219 if (!this->emitPopPtr(E))
8232template <
class Emitter>
8233unsigned Compiler<Emitter>::countHLSLFlatElements(
QualType Ty) {
8236 return VT->getNumElements();
8238 return MT->getNumElementsFlattened();
8242 return CAT->getZExtSize() * countHLSLFlatElements(CAT->getElementType());
8246 const Record *
R = getRecord(Ty);
8250 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8252 Count += countHLSLFlatElements(BS.getType());
8254 for (
const Record::Field &F :
R->fields()) {
8255 if (F.isUnnamedBitField())
8257 Count += countHLSLFlatElements(F.Decl->getType());
8262 if (canClassify(Ty))
8271template <
class Emitter>
8272bool Compiler<Emitter>::emitHLSLFlattenAggregate(
8273 QualType SrcType,
unsigned SrcOffset,
8278 auto saveToLocal = [&](
PrimType T) ->
bool {
8279 unsigned Offset = allocateLocalPrimitive(E, T,
true);
8280 if (!this->emitSetLocal(T, Offset, E))
8282 Elements.push_back({Offset, T});
8288 unsigned Offset = allocateLocalPrimitive(E,
PT_Ptr,
true);
8289 if (!this->emitSetLocal(
PT_Ptr, Offset, E))
8290 return std::nullopt;
8295 unsigned NumElems = 0;
8298 NumElems = VT->getNumElements();
8299 ElemType = VT->getElementType();
8301 NumElems = MT->getNumElementsFlattened();
8302 ElemType = MT->getElementType();
8305 PrimType ElemT = classifyPrim(ElemType);
8306 for (
unsigned I = 0; I != NumElems && Elements.size() < MaxElements; ++I) {
8307 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8309 if (!this->emitArrayElemPop(ElemT, I, E))
8311 if (!saveToLocal(ElemT))
8321 QualType ArrElemType = CAT->getElementType();
8322 unsigned ArrSize = CAT->getZExtSize();
8325 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8326 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8328 if (!this->emitArrayElemPop(*ElemT, I, E))
8330 if (!saveToLocal(*ElemT))
8334 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8335 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8337 if (!this->emitConstUint32(I, E))
8339 if (!this->emitArrayElemPtrPopUint32(E))
8344 if (!emitHLSLFlattenAggregate(ArrElemType, *ElemPtrOffset, Elements,
8355 const Record *
R = getRecord(SrcType);
8359 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8361 if (Elements.size() >= MaxElements)
8363 const Record::Base *B =
R->getBase(BS.getType());
8365 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8367 if (!this->emitGetPtrBasePop(B->Offset,
false, E))
8372 if (!emitHLSLFlattenAggregate(BS.getType(), *BasePtrOffset, Elements,
8378 for (
const Record::Field &F :
R->fields()) {
8379 if (Elements.size() >= MaxElements)
8381 if (F.isUnnamedBitField())
8384 QualType FieldType = F.Decl->getType();
8385 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8387 if (!this->emitGetPtrFieldPop(F.Offset, E))
8391 if (!this->emitLoadPop(*FieldT, E))
8393 if (!saveToLocal(*FieldT))
8397 if (!FieldPtrOffset)
8399 if (!emitHLSLFlattenAggregate(FieldType, *FieldPtrOffset, Elements,
8415template <
class Emitter>
8416bool Compiler<Emitter>::emitHLSLConstructAggregate(
8422 const auto &Src = Elements[ElemIdx++];
8423 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
8425 return this->emitPrimCast(Src.Type, DestT, DestQT, E);
8429 unsigned NumElems = 0;
8432 NumElems = VT->getNumElements();
8433 ElemType = VT->getElementType();
8435 NumElems = MT->getNumElementsFlattened();
8436 ElemType = MT->getElementType();
8439 PrimType DestElemT = classifyPrim(ElemType);
8440 for (
unsigned I = 0; I != NumElems; ++I) {
8441 if (!loadAndCast(DestElemT, ElemType))
8443 if (!this->emitInitElem(DestElemT, I, E))
8453 QualType ArrElemType = CAT->getElementType();
8454 unsigned ArrSize = CAT->getZExtSize();
8457 for (
unsigned I = 0; I != ArrSize; ++I) {
8458 if (!loadAndCast(*ElemT, ArrElemType))
8460 if (!this->emitInitElem(*ElemT, I, E))
8464 for (
unsigned I = 0; I != ArrSize; ++I) {
8465 if (!this->emitConstUint32(I, E))
8467 if (!this->emitArrayElemPtrUint32(E))
8469 if (!emitHLSLConstructAggregate(ArrElemType, Elements, ElemIdx, E))
8471 if (!this->emitFinishInitPop(E))
8481 const Record *
R = getRecord(DestType);
8485 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8487 const Record::Base *B =
R->getBase(BS.getType());
8489 if (!this->emitGetPtrBase(B->Offset, E))
8491 if (!emitHLSLConstructAggregate(BS.getType(), Elements, ElemIdx, E))
8493 if (!this->emitFinishInitPop(E))
8498 for (
const Record::Field &F :
R->fields()) {
8499 if (F.isUnnamedBitField())
8502 QualType FieldType = F.Decl->getType();
8504 if (!loadAndCast(*FieldT, FieldType))
8506 if (F.isBitField()) {
8507 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8510 if (!this->emitInitField(*FieldT, F.Offset, E))
8514 if (!this->emitGetPtrField(F.Offset, E))
8516 if (!emitHLSLConstructAggregate(FieldType, Elements, ElemIdx, E))
8518 if (!this->emitPopPtr(E))
static void emit(Program &P, llvm::SmallVectorImpl< std::byte > &Code, const T &Val, bool &Success)
Helper to write bytecode and bail out if 32-bit offsets become invalid.
static void emitCleanup(CIRGenFunction &cgf, cir::CleanupScopeOp cleanupScope, EHScopeStack::Cleanup *cleanup, EHScopeStack::Cleanup::Flags flags, Address activeFlag)
static uint32_t getBitWidth(const Expr *E)
#define EMIT_ARITH_OP(OP)
static CharUnits AlignOfType(QualType T, const ASTContext &ASTCtx, UnaryExprOrTypeTrait Kind)
static const Expr * stripDerivedToBaseCasts(const Expr *E)
static const Expr * stripCheckedDerivedToBaseCasts(const Expr *E)
static bool hasTrivialDefaultCtorParent(const FieldDecl *FD)
static bool initNeedsOverridenLoc(const CXXCtorInitializer *Init)
Result
Implement __builtin_bit_cast and related operations.
a trap message and trap category.
llvm::APInt getValue() const
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
const LValueBase getLValueBase() const
APValue & getArrayInitializedElt(unsigned I)
ArrayRef< LValuePathEntry > getLValuePath() const
APValue & getStructField(unsigned i)
const FieldDecl * getUnionField() const
unsigned getStructNumFields() const
bool isMemberPointerToDerivedMember() const
unsigned getStructNumBases() const
const ValueDecl * getMemberPointerDecl() const
APValue & getUnionValue()
ArrayRef< const CXXRecordDecl * > getMemberPointerPath() const
bool isMemberPointer() const
unsigned getArraySize() const
@ None
There is no such object (it's outside its lifetime).
bool isNullPointer() const
APValue & getStructBase(unsigned i)
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.
unsigned getPreferredTypeAlign(QualType T) const
Return the "preferred" alignment of the specified type T for the current target, in bits.
const LangOptions & getLangOpts() const
unsigned getOpenMPDefaultSimdAlign(QualType T) const
Get default simd alignment of the specified complete type in bits.
TypeInfoChars getTypeInfoDataSizeInChars(QualType T) const
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
llvm::FixedPointSemantics getFixedPointSemantics(QualType Ty) const
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.
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...
AddrLabelExpr - The GNU address of label extension, representing &&label.
Represents the index of the current element of an array being initialized by an ArrayInitLoopExpr.
Represents a loop initializing the elements of an array.
llvm::APInt getArraySize() const
OpaqueValueExpr * getCommonExpr() const
Get the common subexpression shared by all initializations (the source array).
Expr * getSubExpr() const
Get the initializer to use for each array element.
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
Expr * getLHS()
An array access can be written A[4] or 4[A] (both are equivalent).
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
uint64_t getValue() const
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Attr - This represents one attribute.
Represents an attribute applied to a statement.
ArrayRef< const Attr * > getAttrs() const
Represents a C++ declaration that introduces decls from somewhere else.
A builtin binary operation expression such as "x + y" or "x <= y".
static bool isLogicalOp(Opcode Opc)
static bool isComparisonOp(Opcode Opc)
static bool isShiftOp(Opcode Opc)
static bool isCommaOp(Opcode Opc)
static Opcode getOpForCompoundAssignment(Opcode Opc)
static bool isPtrMemOp(Opcode Opc)
predicates to categorize the respective opcodes.
static bool isAssignmentOp(Opcode Opc)
static bool isCompoundAssignmentOp(Opcode Opc)
static bool isBitwiseOp(Opcode Opc)
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
BreakStmt - This represents a break.
Represents a C++2a __builtin_bit_cast(T, v) expression.
Represents a base class of a C++ class.
Represents binding an expression to a temporary.
const Expr * getSubExpr() const
A boolean literal, per ([C++ lex.bool] Boolean literals).
Represents a call to a C++ constructor.
bool isElidable() const
Whether this construction is elidable.
Expr * getArg(unsigned Arg)
Return the specified argument.
bool requiresZeroInitialization() const
Whether this construction first requires zero-initialization before the initializer is called.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will (ultimately) call.
unsigned getNumArgs() const
Return the number of arguments to the constructor call.
Represents a C++ constructor within a class.
bool isCopyOrMoveConstructor(unsigned &TypeQuals) const
Determine whether this is a copy or move constructor.
Represents a C++ base or member initializer.
A default argument (C++ [dcl.fct.default]).
A use of a default initializer in a constructor or in aggregate initialization.
Expr * getExpr()
Get the initialization expression that will be used.
Represents a delete expression for memory deallocation and destructor calls, e.g.
FunctionDecl * getOperatorDelete() const
bool isGlobalDelete() const
Represents a C++ destructor within a class.
A C++ dynamic_cast expression (C++ [expr.dynamic.cast]).
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
DeclStmt * getBeginStmt()
DeclStmt * getLoopVarStmt()
DeclStmt * getRangeStmt()
Represents a call to an inherited base class constructor from an inheriting constructor.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will call.
Represents a static or instance method of a struct/union/class.
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
bool isMoveAssignmentOperator() const
Determine whether this is a move assignment operator.
bool isCopyAssignmentOperator() const
Determine whether this is a copy-assignment operator, regardless of whether it was declared implicitl...
bool isLambdaStaticInvoker() const
Determine whether this is a lambda closure type's static member function that is used for the result ...
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
QualType getAllocatedType() const
std::optional< Expr * > getArraySize()
This might return std::nullopt even if isArray() returns true, since there might not be an array size...
Expr * getPlacementArg(unsigned I)
unsigned getNumPlacementArgs() const
FunctionDecl * getOperatorNew() const
Expr * getInitializer()
The initializer of this new-expression.
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
The null pointer literal (C++11 [lex.nullptr])
Represents a list-initialization with parenthesis.
MutableArrayRef< Expr * > getInitExprs()
Represents a C++ struct/union/class.
bool hasTrivialDefaultConstructor() const
Determine whether this class has a trivial default constructor (C++11 [class.ctor]p5).
bool isGenericLambda() const
Determine whether this class describes a generic lambda function object (i.e.
capture_const_range captures() const
CXXMethodDecl * getLambdaCallOperator() const
Retrieve the lambda call operator of the closure type if this is a closure type.
A C++ reinterpret_cast expression (C++ [expr.reinterpret.cast]).
A rewritten comparison expression that was originally written using operator syntax.
Expr * getSemanticForm()
Get an equivalent semantic form for this expression.
An expression "T()" which creates an rvalue of a non-class type T.
Implicit construction of a std::initializer_list<T> object from an array temporary within list-initia...
Represents the this expression in C++.
A C++ throw-expression (C++ [except.throw]).
const Expr * getSubExpr() const
CXXTryStmt - A C++ try block, including all handlers.
CompoundStmt * getTryBlock()
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
bool isTypeOperand() const
QualType getTypeOperand(const ASTContext &Context) const
Retrieves the type operand of this typeid() expression after various required adjustments (removing r...
Expr * getExprOperand() const
bool isPotentiallyEvaluated() const
Determine whether this typeid has a type operand which is potentially evaluated, per C++11 [expr....
A Microsoft C++ __uuidof expression, which gets the _GUID that corresponds to the supplied type or ex...
MSGuidDecl * getGuidDecl() const
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
FunctionDecl * getDirectCallee()
If the callee is a FunctionDecl, return it. Otherwise return null.
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this call.
Expr ** getArgs()
Retrieve the call arguments.
QualType getCallReturnType(const ASTContext &Ctx) const
getCallReturnType - Get the return type of the call expr.
CaseStmt - Represent a case statement.
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
path_iterator path_begin()
CastKind getCastKind() const
llvm::iterator_range< path_iterator > path()
Path through the class hierarchy taken by casts between base and derived classes (see implementation ...
const FieldDecl * getTargetUnionField() const
CharUnits - This is an opaque type for sizes expressed in character units.
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.
unsigned getValue() const
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Expr * getChosenSubExpr() const
getChosenSubExpr - Return the subexpression chosen according to the condition.
Complex values, per C99 6.2.5p11.
QualType getElementType() const
CompoundAssignOperator - For compound assignments (e.g.
QualType getComputationLHSType() const
QualType getComputationResultType() const
CompoundLiteralExpr - [C99 6.5.2.5].
const Expr * getInitializer() const
CompoundStmt - This represents a group of statements like { stmt stmt }.
Represents the specialization of a concept - evaluates to a prvalue of type bool.
bool isSatisfied() const
Whether or not the concept with the given arguments was satisfied when the expression was created.
Represents the canonical version of C arrays with a specified constant size.
uint64_t getZExtSize() const
Return the size zero-extended as a uint64_t.
ConstantExpr - An expression that occurs in a constant context and optionally the result of evaluatin...
APValue getAPValueResult() const
bool hasAPValueResult() const
Represents a concrete matrix type with constant number of rows and columns.
ContinueStmt - This represents a continue.
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
Expr * getSrcExpr() const
getSrcExpr - Return the Expr to be converted.
DeclContext * getParent()
getParent - Returns the containing DeclContext.
A reference to a declared variable, function, enum, etc.
DeclStmt - Adaptor class for mixing declarations with statements and expressions.
Decl - This represents one declaration (or definition), e.g.
bool isInvalidDecl() const
OverloadedOperatorKind getCXXOverloadedOperator() const
If this name is the name of an overloadable operator in C++ (e.g., operator+), retrieve the kind of o...
bool isAnyOperatorNew() const
DoStmt - This represents a 'do/while' stmt.
const Expr * getBase() const
Represents a reference to emded data.
ChildElementIter< false > begin()
Represents an expression – generally a full-expression – that introduces cleanups to be run at the en...
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...
bool isValueDependent() const
Determines whether the value of this expression depends on.
Expr * IgnoreImplicit() LLVM_READONLY
Skip past any implicit AST nodes which might surround this expression until reaching a fixed point.
bool containsErrors() const
Whether this expression contains subexpressions which had errors.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
bool isLValue() const
isLValue - True if this expression is an "l-value" according to the rules of the current language.
bool HasSideEffects(const ASTContext &Ctx, bool IncludePossibleEffects=true) const
HasSideEffects - This routine returns true for all those expressions which have any effect other than...
bool isTemporaryObject(ASTContext &Ctx, const CXXRecordDecl *TempTy) const
Determine whether the result of this expression is a temporary object of the given class type.
bool refersToBitField() const
Returns true if this expression is a gl-value that potentially refers to a bit-field.
An expression trait intrinsic.
ExtVectorElementExpr - This represents access to specific elements of a vector, and may occur on the ...
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.
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
llvm::APInt getValue() const
Returns an internal integer representation of the literal.
llvm::APFloat getValue() const
ForStmt - This represents a 'for (init;cond;inc)' stmt.
VarDecl * getConditionVariable() const
Retrieve the variable declared in this "for" statement, if any.
DeclStmt * getConditionVariableDeclStmt()
If this ForStmt has a condition variable, return the faux DeclStmt associated with the creation of th...
const Expr * getSubExpr() const
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
bool isFunctionTemplateSpecialization() const
Determine whether this function is a function template specialization.
FunctionTemplateDecl * getDescribedFunctionTemplate() const
Retrieves the function template that is described by this function declaration.
unsigned getBuiltinID(bool ConsiderWrapperFunctions=false) const
Returns a value indicating whether this function corresponds to a builtin function.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
const TemplateArgumentList * getTemplateSpecializationArgs() const
Retrieve the template arguments used to produce this function template specialization from the primar...
bool isUsableAsGlobalAllocationFunctionInConstantEvaluation(UnsignedOrNone *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions described in i...
bool isDefaulted() const
Whether this function is defaulted.
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Declaration of a template function.
FunctionDecl * findSpecialization(ArrayRef< TemplateArgument > Args, void *&InsertPos)
Return the specialization with the provided arguments if it exists, otherwise return the insertion po...
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
Expr * getResultExpr()
Return the result expression of this controlling expression.
IfStmt - This represents an if/then/else.
bool isNonNegatedConsteval() const
bool isNegatedConsteval() const
DeclStmt * getConditionVariableDeclStmt()
If this IfStmt has a condition variable, return the faux DeclStmt associated with the creation of tha...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "if" statement, if any.
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
const Expr * getSubExpr() const
Represents an implicitly-generated value initialization of an object of a given type.
Represents a field injected from an anonymous union/struct into the parent scope.
Describes an C or C++ initializer list.
Expr * getArrayFiller()
If this initializer list initializes an array with more elements than there are initializers in the l...
ArrayRef< Expr * > inits()
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
capture_init_iterator capture_init_begin()
Retrieve the first initialization argument for this lambda expression (which initializes the first ca...
CXXRecordDecl * getLambdaClass() const
Retrieve the class that corresponds to the lambda.
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
const Stmt * getNamedLoopOrSwitch() const
If this is a named break/continue, get the loop or switch statement that this targets.
APValue & getAsAPValue() const
Get the value of this MSGuidDecl as an APValue.
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.
LifetimeExtendedTemporaryDecl * getLifetimeExtendedTemporaryDecl()
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
ValueDecl * getMemberDecl() const
Retrieve the member declaration to which this expression refers.
A pointer to member type per C++ 8.3.3 - Pointers to members.
This represents a decl that may have a name.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
Represents a C++ namespace alias.
ObjCArrayLiteral - used for objective-c array containers; as in: @["Hello", NSApp,...
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
ObjCDictionaryLiteral - AST node to represent objective-c dictionary literals; as in:"name" : NSUserN...
ObjCEncodeExpr, used for @encode in Objective-C.
QualType getEncodedType() const
SourceLocation getAtLoc() const
bool isExpressibleAsConstantInitializer() const
ObjCStringLiteral, used for Objective-C string literals i.e.
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Expr * getIndexExpr(unsigned Idx)
const OffsetOfNode & getComponent(unsigned Idx) const
unsigned getNumComponents() const
Helper class for OffsetOfExpr.
unsigned getArrayExprIndex() const
For an array element node, returns the index into the array of expressions.
@ Array
An index into an array.
Kind getKind() const
Determine what kind of offsetof node this is.
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 ...
Expr * getSelectedExpr() const
ParenExpr - This represents a parenthesized expression, e.g.
const Expr * getSubExpr() const
Represents a parameter to a function.
PointerType - C99 6.7.5.1 - Pointer Declarators.
QualType getPointeeType() const
[C99 6.4.2.2] - A predefined identifier such as func.
StringLiteral * getFunctionName()
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 isVolatileQualified() const
Determine whether this type is volatile-qualified.
QualType withConst() const
bool isNull() const
Return true if this QualType doesn't point to a type yet.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
bool isConstant(const ASTContext &Ctx) const
bool isConstQualified() const
Determine whether this type is const-qualified.
Represents a struct/union/class.
Frontend produces RecoveryExprs on semantic errors that prevent creating other well-formed expression...
Base for LValueReferenceType and RValueReferenceType.
C++2a [expr.prim.req]: A requires-expression provides a concise way to express requirements on templa...
bool isSatisfied() const
Whether or not the requires clause is satisfied.
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
SourceLocation getLocation() const
std::string ComputeName(ASTContext &Context) const
Scope - A scope is a transient data structure that is used while parsing the program.
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
llvm::APSInt getShuffleMaskIdx(unsigned N) const
unsigned getNumSubExprs() const
getNumSubExprs - Return the size of the SubExprs array.
Expr * getExpr(unsigned Index)
getExpr - Return the Expr at the specified index.
Represents an expression that computes the length of a parameter pack.
unsigned getPackLength() const
Retrieve the length of the parameter pack.
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
APValue EvaluateInContext(const ASTContext &Ctx, const Expr *DefaultExpr) const
Return the result of evaluating this SourceLocExpr in the specified (and possibly null) default argum...
Represents a C++11 static_assert declaration.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
CompoundStmt * getSubStmt()
bool Visit(PTR(Stmt) S, ParamTys... P)
Stmt - This represents one statement.
StmtClass getStmtClass() const
StringLiteral - This represents a string literal expression, e.g.
unsigned getLength() const
static StringLiteral * Create(const ASTContext &Ctx, StringRef Str, StringLiteralKind Kind, bool Pascal, QualType Ty, ArrayRef< SourceLocation > Locs)
This is the "fully general" constructor that allows representation of strings formed from one or more...
uint32_t getCodeUnit(size_t i) const
unsigned getCharByteWidth() const
Represents a reference to a non-type template parameter that has been substituted with a template arg...
Expr * getReplacement() const
const SwitchCase * getNextSwitchCase() const
SwitchStmt - This represents a 'switch' stmt.
VarDecl * getConditionVariable()
Retrieve the variable declared in this "switch" statement, if any.
SwitchCase * getSwitchCaseList()
DeclStmt * getConditionVariableDeclStmt()
If this SwitchStmt has a condition variable, return the faux DeclStmt associated with the creation of...
Represents the declaration of a struct/union/class/enum.
bool isCompleteDefinition() const
Return true if this decl has its body fully specified.
A template argument list.
ArrayRef< TemplateArgument > asArray() const
Produce this as an array ref.
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
bool getBoolValue() const
const APValue & getAPValue() const
bool isStoredAsBoolean() const
The base class of the type hierarchy.
bool isBooleanType() const
bool isLiteralType(const ASTContext &Ctx) const
Return true if this is a literal type (C++11 [basic.types]p10)
bool isIncompleteArrayType() const
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isVoidPointerType() const
bool isConstantSizeType() const
Return true if this is not a variable sized type, according to the rules of C99 6....
bool isFunctionPointerType() const
bool isConstantMatrixType() const
bool isPointerType() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
bool isEnumeralType() const
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isIntegralOrEnumerationType() const
Determine whether this type is an integral or enumeration type.
bool isSpecificBuiltinType(unsigned K) const
Test for a particular builtin type.
bool isDependentType() const
Whether this type is a dependent type, meaning that its definition somehow depends on a template para...
bool isAnyComplexType() const
bool isFixedPointType() const
Return true if this is a fixed point type according to ISO/IEC JTC1 SC22 WG14 N1169.
const Type * getBaseElementTypeUnsafe() const
Get the base element type of this type, potentially discarding type qualifiers.
bool isMemberPointerType() const
bool isAtomicType() const
EnumDecl * castAsEnumDecl() const
bool isStdByteType() const
const ArrayType * getAsArrayTypeUnsafe() const
A variant of getAs<> for array types which silently discards qualifiers from the outermost type.
bool isPointerOrReferenceType() const
bool isFunctionType() const
bool isVectorType() const
bool isRealFloatingType() const
Floating point categories.
const T * getAsCanonical() const
If this type is canonically the specified type, return its canonical type cast to that specified type...
bool isFloatingType() const
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
bool isSizelessVectorType() const
Returns true for all scalable vector types.
Base class for declarations which introduce a typedef-name.
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
QualType getArgumentType() const
QualType getTypeOfArgument() const
Gets the argument type, or the type of the argument expression, whichever is appropriate.
bool isArgumentType() const
UnaryExprOrTypeTrait getKind() const
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Expr * getSubExpr() const
bool canOverflow() const
Returns true if the unary operator can cause an overflow.
Represents C++ using-directive.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
bool isWeak() const
Determine whether this symbol is weakly-imported, or declared with the weak or weak-ref attr.
Represents a variable declaration or definition.
bool isInitCapture() const
Whether this variable is the implicit variable for a lambda init-capture.
APValue * evaluateValue() const
Attempt to evaluate the value of the initializer attached to this declaration, and produce notes expl...
bool isStaticDataMember() const
Determines whether this is a static data member.
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
bool hasConstantInitialization() const
Determine whether this variable has constant initialization.
bool isStaticLocal() const
Returns true if a variable with function scope is a static local variable.
const Expr * getInit() const
bool isLocalVarDecl() const
Returns true for local variable declarations other than parameters.
const Expr * getAnyInitializer() const
Get the initializer for this variable, no matter which declaration it is attached to.
Represents a GCC generic vector type.
unsigned getNumElements() const
QualType getElementType() const
WhileStmt - This represents a 'while' stmt.
DeclStmt * getConditionVariableDeclStmt()
If this WhileStmt has a condition variable, return the faux DeclStmt associated with the creation of ...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "while" statement, if any.
A memory block, either on the stack or in the heap.
void invokeDtor()
Invokes the Destructor.
Compilation context for expressions.
llvm::SmallVector< InitLink > InitStack
bool VisitArrayInitIndexExpr(const ArrayInitIndexExpr *E)
bool VisitCXXDeleteExpr(const CXXDeleteExpr *E)
bool VisitOffsetOfExpr(const OffsetOfExpr *E)
bool visitContinueStmt(const ContinueStmt *S)
bool VisitCharacterLiteral(const CharacterLiteral *E)
bool visitArrayElemInit(unsigned ElemIndex, const Expr *Init, OptPrimType InitT)
Pointer to the array(not the element!) must be on the stack when calling this.
UnsignedOrNone allocateLocal(DeclTy &&Decl, QualType Ty=QualType(), ScopeKind=ScopeKind::Block)
Allocates a space storing a local given its type.
bool VisitCXXParenListInitExpr(const CXXParenListInitExpr *E)
bool VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E)
bool visitInitializerPop(const Expr *E)
Similar, but will also pop the pointer.
bool VisitCompoundLiteralExpr(const CompoundLiteralExpr *E)
bool visitBool(const Expr *E)
Visits an expression and converts it to a boolean.
bool VisitCXXDefaultInitExpr(const CXXDefaultInitExpr *E)
PrimType classifyPrim(QualType Ty) const
Classifies a known primitive type.
bool VisitTypeTraitExpr(const TypeTraitExpr *E)
bool VisitLambdaExpr(const LambdaExpr *E)
bool VisitMemberExpr(const MemberExpr *E)
llvm::DenseMap< const OpaqueValueExpr *, unsigned > OpaqueExprs
OpaqueValueExpr to location mapping.
bool VisitBinaryOperator(const BinaryOperator *E)
bool visitAttributedStmt(const AttributedStmt *S)
bool VisitPackIndexingExpr(const PackIndexingExpr *E)
bool visitAPValueInitializer(const APValue &Val, const Expr *E, QualType T)
bool VisitArraySubscriptExpr(const ArraySubscriptExpr *E)
bool VisitCallExpr(const CallExpr *E)
std::optional< uint64_t > ArrayIndex
Current argument index. Needed to emit ArrayInitIndexExpr.
bool VisitPseudoObjectExpr(const PseudoObjectExpr *E)
bool VisitCXXReinterpretCastExpr(const CXXReinterpretCastExpr *E)
const Function * getFunction(const FunctionDecl *FD)
Returns a function for the given FunctionDecl.
bool VisitFixedPointBinOp(const BinaryOperator *E)
bool VisitCastExpr(const CastExpr *E)
bool VisitObjCEncodeExpr(const ObjCEncodeExpr *E)
bool VisitFixedPointUnaryOperator(const UnaryOperator *E)
unsigned allocateLocalPrimitive(DeclTy &&Decl, PrimType Ty, bool IsConst, bool IsVolatile=false, ScopeKind SC=ScopeKind::Block)
Creates a local primitive value.
bool VisitComplexUnaryOperator(const UnaryOperator *E)
llvm::DenseMap< const SwitchCase *, LabelTy > CaseMap
bool VisitBlockExpr(const BlockExpr *E)
bool visitAPValue(const APValue &Val, PrimType ValType, const Expr *E)
Visit an APValue.
bool VisitCXXScalarValueInitExpr(const CXXScalarValueInitExpr *E)
bool VisitLogicalBinOp(const BinaryOperator *E)
bool visitCompoundStmt(const CompoundStmt *S)
Context & Ctx
Current compilation context.
bool visitDeclRef(const ValueDecl *D, const Expr *E)
Visit the given decl as if we have a reference to it.
bool visitBreakStmt(const BreakStmt *S)
bool visitExpr(const Expr *E, bool DestroyToplevelScope) override
bool visitForStmt(const ForStmt *S)
bool VisitDeclRefExpr(const DeclRefExpr *E)
bool VisitOpaqueValueExpr(const OpaqueValueExpr *E)
bool VisitArrayInitLoopExpr(const ArrayInitLoopExpr *E)
bool VisitStmtExpr(const StmtExpr *E)
bool VisitBuiltinBitCastExpr(const BuiltinBitCastExpr *E)
bool VisitFixedPointLiteral(const FixedPointLiteral *E)
const FunctionDecl * CompilingFunction
bool VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *E)
VarCreationState visitVarDecl(const VarDecl *VD, const Expr *Init, bool Toplevel=false)
Creates and initializes a variable from the given decl.
VariableScope< Emitter > * VarScope
Current scope.
bool visitDeclAndReturn(const VarDecl *VD, const Expr *Init, bool ConstantContext) override
Toplevel visitDeclAndReturn().
bool VisitCXXNewExpr(const CXXNewExpr *E)
const ValueDecl * InitializingDecl
bool VisitCompoundAssignOperator(const CompoundAssignOperator *E)
bool visit(const Expr *E) override
Evaluates an expression and places the result on the stack.
bool delegate(const Expr *E)
Just pass evaluation on to E.
bool discard(const Expr *E)
Evaluates an expression for side effects and discards the result.
bool VisitCXXDefaultArgExpr(const CXXDefaultArgExpr *E)
CaseMap CaseLabels
Switch case mapping.
Record * getRecord(QualType Ty)
Returns a record from a record or pointer type.
const RecordType * getRecordTy(QualType Ty)
Returns a record type from a record or pointer type.
bool VisitCXXStdInitializerListExpr(const CXXStdInitializerListExpr *E)
bool visitInitList(ArrayRef< const Expr * > Inits, const Expr *ArrayFiller, const Expr *E)
bool VisitSizeOfPackExpr(const SizeOfPackExpr *E)
bool VisitPredefinedExpr(const PredefinedExpr *E)
bool VisitSourceLocExpr(const SourceLocExpr *E)
bool visitDeclStmt(const DeclStmt *DS, bool EvaluateConditionDecl=false)
bool emitCleanup()
Emits scope cleanup instructions.
bool VisitExtVectorElementExpr(const ExtVectorElementExpr *E)
bool VisitObjCStringLiteral(const ObjCStringLiteral *E)
bool VisitArrayTypeTraitExpr(const ArrayTypeTraitExpr *E)
bool visitInitializer(const Expr *E)
Compiles an initializer.
const Expr * SourceLocDefaultExpr
DefaultInit- or DefaultArgExpr, needed for SourceLocExpr.
bool VisitObjCArrayLiteral(const ObjCArrayLiteral *E)
UnsignedOrNone OptLabelTy
bool VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *E)
bool VisitPointerArithBinOp(const BinaryOperator *E)
Perform addition/subtraction of a pointer and an integer or subtraction of two pointers.
bool visitCallArgs(ArrayRef< const Expr * > Args, const FunctionDecl *FuncDecl, bool Activate, bool IsOperatorCall)
bool VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *E)
bool visitDefaultStmt(const DefaultStmt *S)
bool VisitObjCDictionaryLiteral(const ObjCDictionaryLiteral *E)
typename Emitter::LabelTy LabelTy
VarCreationState visitDecl(const VarDecl *VD)
bool VisitCXXDynamicCastExpr(const CXXDynamicCastExpr *E)
bool visitStmt(const Stmt *S)
bool VisitExpressionTraitExpr(const ExpressionTraitExpr *E)
bool VisitVectorUnaryOperator(const UnaryOperator *E)
bool VisitCXXConstructExpr(const CXXConstructExpr *E)
bool VisitCXXNullPtrLiteralExpr(const CXXNullPtrLiteralExpr *E)
bool VisitObjCBoxedExpr(const ObjCBoxedExpr *E)
bool VisitCXXInheritedCtorInitExpr(const CXXInheritedCtorInitExpr *E)
bool VisitRecoveryExpr(const RecoveryExpr *E)
bool VisitRequiresExpr(const RequiresExpr *E)
bool Initializing
Flag inidicating if we're initializing an already created variable.
bool visitReturnStmt(const ReturnStmt *RS)
bool VisitCXXThrowExpr(const CXXThrowExpr *E)
bool VisitSubstNonTypeTemplateParmExpr(const SubstNonTypeTemplateParmExpr *E)
bool VisitChooseExpr(const ChooseExpr *E)
bool visitFunc(const FunctionDecl *F) override
bool visitCXXForRangeStmt(const CXXForRangeStmt *S)
bool visitCaseStmt(const CaseStmt *S)
bool VisitComplexBinOp(const BinaryOperator *E)
llvm::DenseMap< const ValueDecl *, Scope::Local > Locals
Variable to storage mapping.
bool VisitAbstractConditionalOperator(const AbstractConditionalOperator *E)
bool VisitCXXTypeidExpr(const CXXTypeidExpr *E)
UnsignedOrNone allocateTemporary(const Expr *E)
bool VisitBuiltinCallExpr(const CallExpr *E, unsigned BuiltinID)
bool VisitImplicitValueInitExpr(const ImplicitValueInitExpr *E)
bool VisitCXXRewrittenBinaryOperator(const CXXRewrittenBinaryOperator *E)
OptPrimType ReturnType
Type of the expression returned by the function.
bool VisitUnaryOperator(const UnaryOperator *E)
bool VisitFloatCompoundAssignOperator(const CompoundAssignOperator *E)
OptPrimType classify(const Expr *E) const
llvm::SmallVector< LabelInfo > LabelInfoStack
Stack of label information for loops and switch statements.
bool VisitGenericSelectionExpr(const GenericSelectionExpr *E)
bool visitDoStmt(const DoStmt *S)
bool VisitIntegerLiteral(const IntegerLiteral *E)
bool VisitInitListExpr(const InitListExpr *E)
bool VisitVectorBinOp(const BinaryOperator *E)
bool VisitStringLiteral(const StringLiteral *E)
bool VisitParenExpr(const ParenExpr *E)
bool VisitCXXNoexceptExpr(const CXXNoexceptExpr *E)
bool VisitShuffleVectorExpr(const ShuffleVectorExpr *E)
bool VisitPointerCompoundAssignOperator(const CompoundAssignOperator *E)
bool DiscardResult
Flag indicating if return value is to be discarded.
bool VisitEmbedExpr(const EmbedExpr *E)
bool VisitConvertVectorExpr(const ConvertVectorExpr *E)
bool VisitCXXThisExpr(const CXXThisExpr *E)
bool VisitConstantExpr(const ConstantExpr *E)
bool VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *E)
bool visitSwitchStmt(const SwitchStmt *S)
bool VisitCXXUuidofExpr(const CXXUuidofExpr *E)
bool VisitExprWithCleanups(const ExprWithCleanups *E)
bool visitAsLValue(const Expr *E)
bool visitWhileStmt(const WhileStmt *S)
bool visitIfStmt(const IfStmt *IS)
bool VisitAddrLabelExpr(const AddrLabelExpr *E)
bool canClassify(const Expr *E) const
bool VisitFloatingLiteral(const FloatingLiteral *E)
Program & P
Program to link to.
bool VisitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *E)
bool VariablesAreConstexprUnknown
bool VisitGNUNullExpr(const GNUNullExpr *E)
bool VisitImaginaryLiteral(const ImaginaryLiteral *E)
bool VisitSYCLUniqueStableNameExpr(const SYCLUniqueStableNameExpr *E)
bool visitCXXTryStmt(const CXXTryStmt *S)
static bool isUnevaluatedBuiltin(unsigned ID)
Unevaluated builtins don't get their arguments put on the stack automatically.
static bool shouldBeGloballyIndexed(const ValueDecl *VD)
Returns whether we should create a global variable for the given ValueDecl.
Scope used to handle temporaries in toplevel variable declarations.
DeclScope(Compiler< Emitter > *Ctx, const ValueDecl *VD)
Wrapper around fixed point types.
static FixedPoint zero(llvm::FixedPointSemantics Sem)
If a Floating is constructed from Memory, it DOES NOT OWN THAT MEMORY.
bool hasThisPointer() const
bool hasRVO() const
Checks if the first argument is a RVO pointer.
When generating code for e.g.
LocOverrideScope(Compiler< Emitter > *Ctx, SourceInfo NewValue, bool Enabled=true)
Generic scope for local variables.
bool destroyLocals(const Expr *E=nullptr) override
Explicit destruction of local variables.
LocalScope(Compiler< Emitter > *Ctx, ScopeKind Kind=ScopeKind::Block)
Sets the context for break/continue statements.
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
LoopScope(Compiler< Emitter > *Ctx, const Stmt *Name, LabelTy BreakLabel, LabelTy ContinueLabel)
PrimType value_or(PrimType PT) const
Scope used to handle initialization methods.
OptionScope(Compiler< Emitter > *Ctx, bool NewDiscardResult, bool NewInitializing, bool NewToLValue)
Root constructor, compiling or discarding primitives.
Context to manage declaration lifetimes.
Structure/Class descriptor.
bool isUnion() const
Checks if the record is a union.
const Field * getField(unsigned I) const
const Base * getBase(unsigned I) const
bool hasTrivialDtor() const
Returns true for anonymous unions and records with no destructor or for those with a trivial destruct...
Describes the statement/declaration an opcode was generated from.
StmtExprScope(Compiler< Emitter > *Ctx)
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
typename Compiler< Emitter >::CaseMap CaseMap
SwitchScope(Compiler< Emitter > *Ctx, const Stmt *Name, CaseMap &&CaseLabels, LabelTy BreakLabel, OptLabelTy DefaultLabel)
Scope chain managing the variable lifetimes.
Compiler< Emitter > * Ctx
Compiler instance.
virtual void addLocal(Scope::Local Local)
VariableScope * getParent() const
bool Sub(InterpState &S, CodePtr OpPC)
bool LT(InterpState &S, CodePtr OpPC)
static llvm::RoundingMode getRoundingMode(FPOptions FPO)
llvm::PointerUnion< const Decl *, const Expr * > DeclTy
constexpr bool isSignedType(PrimType T)
bool Div(InterpState &S, CodePtr OpPC)
1) Pops the RHS from the stack.
static bool Activate(InterpState &S, CodePtr OpPC)
constexpr bool isPtrType(PrimType T)
constexpr size_t align(size_t Size)
Aligns a size to the pointer alignment.
bool InitScope(InterpState &S, CodePtr OpPC, uint32_t I)
constexpr bool isIntegerOrBoolType(PrimType T)
static void discard(InterpStack &Stk, PrimType T)
bool LE(InterpState &S, CodePtr OpPC)
PrimType
Enumeration of the primitive types of the VM.
static std::optional< bool > getBoolValue(const Expr *E)
bool Init(InterpState &S, CodePtr OpPC)
bool Mul(InterpState &S, CodePtr OpPC)
size_t primSize(PrimType Type)
Returns the size of a primitive type in bytes.
bool Inc(InterpState &S, CodePtr OpPC, bool CanOverflow)
1) Pops a pointer from the stack 2) Load the value from the pointer 3) Writes the value increased by ...
bool Add(InterpState &S, CodePtr OpPC)
llvm::BitVector collectNonNullArgs(const FunctionDecl *F, ArrayRef< const Expr * > Args)
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
bool hasSpecificAttr(const Container &container)
@ Success
Annotation was successful.
UnaryExprOrTypeTrait
Names for the "expression or type" traits.
@ SD_Static
Static storage duration.
@ SD_FullExpression
Full-expression storage duration (for temporaries).
@ Result
The result type of a method or function.
OptionalUnsigned< unsigned > UnsignedOrNone
U cast(CodeGen::Address addr)
int const char * function
Describes a memory block created by an allocation site.
unsigned getNumElems() const
Returns the number of elements stored in the block.
bool isPrimitive() const
Checks if the descriptor is of a primitive.
QualType getElemQualType() const
bool hasTrivialDtor() const
Whether variables of this descriptor need their destructor called or not.
bool isCompositeArray() const
Checks if the descriptor is of an array of composites.
const Descriptor *const ElemDesc
Descriptor of the array element.
static constexpr MetadataSize InlineDescMD
bool isPrimitiveArray() const
Checks if the descriptor is of an array of primitives.
PrimType getPrimType() const
bool isRecord() const
Checks if the descriptor is of a record.
const Record *const ElemRecord
Pointer to the record, if block contains records.
bool isArray() const
Checks if the descriptor is of an array.
Descriptor used for global variables.
GlobalInitState InitState
static InitLink InitList()
static InitLink Elem(unsigned Index)
bool emit(Compiler< Emitter > *Ctx, const Expr *E) const
static InitLink Field(unsigned Offset)
static InitLink Decl(const ValueDecl *D)
static InitLink Temp(unsigned Offset)
Information about a local's storage.
State encapsulating if a the variable creation has been successful, unsuccessful, or no variable has ...
static VarCreationState NotCreated()