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);
2209 Ctx.getASTContext().getAsConstantArrayType(QT);
2212 if (
Initializing && !this->emitCheckArrayDestSize(NumElems, E))
2215 if (
Inits.size() == 1 && QT ==
Inits[0]->getType())
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);
2670 if (
const auto *MD = dyn_cast<CXXMethodDecl>(
Member);
2671 MD && !MD->isStatic()) {
2675 if (!this->
discard(Base) && !this->emitSideEffect(E))
2690 const Record::Field *F = R->getField(FD);
2694 const auto maybeLoadValue = [&]() ->
bool {
2698 return this->emitLoadPop(*T, E);
2703 if (F->Decl->getType()->isReferenceType())
2704 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2705 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2708template <
class Emitter>
2718template <
class Emitter>
2732 if (!this->
visit(Common))
2734 return this->emitCopyArray(*SubExprT, 0, 0, Size, E);
2748 for (
size_t I = 0; I != Size; ++I) {
2760template <
class Emitter>
2775 return this->emitGetLocal(SubExprT, It->second, E);
2778 if (!this->
visit(SourceExpr))
2785 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2794 return this->emitGetLocal(SubExprT, LocalIndex, E);
2798template <
class Emitter>
2821 bool IsBcpCall =
false;
2822 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2823 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2827 LabelTy LabelEnd = this->getLabel();
2828 LabelTy LabelFalse = this->getLabel();
2831 if (!this->emitPushIgnoreDiags(E))
2839 if (this->checkingForUndefinedBehavior()) {
2842 if (!this->
discard(FalseExpr))
2848 if (!this->jumpFalse(LabelFalse, E))
2853 if (!this->jump(LabelEnd, E))
2855 this->emitLabel(LabelFalse);
2859 this->fallthrough(LabelEnd);
2860 this->emitLabel(LabelEnd);
2863 return this->emitPopIgnoreDiags(E);
2867template <
class Emitter>
2873 unsigned StringIndex =
P.createGlobalString(E);
2874 return this->emitGetPtrGlobal(StringIndex, E);
2879 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2880 assert(CAT &&
"a string literal that's not a constant array?");
2885 unsigned N = std::min(ArraySize, E->
getLength());
2888 for (
unsigned I = 0; I != N; ++I) {
2891 if (CharWidth == 1) {
2892 this->emitConstSint8(CodeUnit, E);
2893 this->emitInitElemSint8(I, E);
2894 }
else if (CharWidth == 2) {
2895 this->emitConstUint16(CodeUnit, E);
2896 this->emitInitElemUint16(I, E);
2897 }
else if (CharWidth == 4) {
2898 this->emitConstUint32(CodeUnit, E);
2899 this->emitInitElemUint32(I, E);
2901 llvm_unreachable(
"unsupported character width");
2906 for (
unsigned I = N; I != ArraySize; ++I) {
2907 if (CharWidth == 1) {
2908 this->emitConstSint8(0, E);
2909 this->emitInitElemSint8(I, E);
2910 }
else if (CharWidth == 2) {
2911 this->emitConstUint16(0, E);
2912 this->emitInitElemUint16(I, E);
2913 }
else if (CharWidth == 4) {
2914 this->emitConstUint32(0, E);
2915 this->emitInitElemUint32(I, E);
2917 llvm_unreachable(
"unsupported character width");
2924template <
class Emitter>
2928 return this->emitDummyPtr(E, E);
2931template <
class Emitter>
2933 auto &A =
Ctx.getASTContext();
2942template <
class Emitter>
2950 auto &A =
Ctx.getASTContext();
2954 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2955 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2962 unsigned StringIndex =
P.createGlobalString(SL);
2963 return this->emitGetPtrGlobal(StringIndex, E);
2966template <
class Emitter>
2970 return this->emitConst(E->
getValue(), E);
2973template <
class Emitter>
2999 if (!this->emitSetLocal(*RT, TempOffset, E))
3005 if (!this->emitLoad(LHST, E))
3009 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
3010 LHSComputationType, E))
3014 if (!this->emitGetLocal(*RT, TempOffset, E))
3019 if (!this->emitAddf(getFPOptions(E), E))
3023 if (!this->emitSubf(getFPOptions(E), E))
3027 if (!this->emitMulf(getFPOptions(E), E))
3031 if (!this->emitDivf(getFPOptions(E), E))
3042 return this->emitStorePop(LHST, E);
3043 return this->emitStore(LHST, E);
3046template <
class Emitter>
3055 if (Op != BO_AddAssign && Op != BO_SubAssign)
3064 if (!this->emitLoad(*
LT, LHS))
3070 if (Op == BO_AddAssign) {
3071 if (!this->emitAddOffset(*RT, E))
3074 if (!this->emitSubOffset(*RT, E))
3079 return this->emitStorePopPtr(E);
3080 return this->emitStorePtr(E);
3083template <
class Emitter>
3096 if (!
Ctx.getLangOpts().CPlusPlus14)
3097 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
3099 if (!
LT || !RT || !ResultT || !LHSComputationT)
3124 if (!this->emitSetLocal(*RT, TempOffset, E))
3131 if (!this->emitLoad(*
LT, E))
3133 if (
LT != LHSComputationT &&
3139 if (!this->emitGetLocal(*RT, TempOffset, E))
3145 if (!this->emitAdd(*LHSComputationT, E))
3149 if (!this->emitSub(*LHSComputationT, E))
3153 if (!this->emitMul(*LHSComputationT, E))
3157 if (!this->emitDiv(*LHSComputationT, E))
3161 if (!this->emitRem(*LHSComputationT, E))
3165 if (!this->emitShl(*LHSComputationT, *RT, E))
3169 if (!this->emitShr(*LHSComputationT, *RT, E))
3173 if (!this->emitBitAnd(*LHSComputationT, E))
3177 if (!this->emitBitXor(*LHSComputationT, E))
3181 if (!this->emitBitOr(*LHSComputationT, E))
3185 llvm_unreachable(
"Unimplemented compound assign operator");
3189 if (ResultT != LHSComputationT &&
3190 !this->emitIntegralCast(*LHSComputationT, *ResultT, E->
getType(), E))
3196 return this->emitStoreBitFieldPop(*ResultT, E);
3197 return this->emitStorePop(*ResultT, E);
3200 return this->emitStoreBitField(*ResultT, E);
3201 return this->emitStore(*ResultT, E);
3204template <
class Emitter>
3212template <
class Emitter>
3227 if (!
Ctx.getLangOpts().CPlusPlus11)
3234 for (
const Expr *LHS : CommaLHSs) {
3256 if (!this->
visit(Inner))
3261 if (!this->emitInitGlobalTemp(*InnerT, *GlobalIndex, TempDecl, E))
3264 if (!this->emitInitGlobal(*InnerT, *GlobalIndex, E))
3267 return this->emitGetPtrGlobal(*GlobalIndex, E);
3270 if (!this->checkLiteralType(Inner))
3273 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3279 return this->emitInitGlobalTempComp(TempDecl, E);
3292 unsigned LocalIndex =
3294 if (!this->VarScope->LocalsAlwaysEnabled &&
3295 !this->emitEnableLocal(LocalIndex, E))
3298 if (!this->
visit(Inner))
3300 if (!this->emitSetLocal(*InnerT, LocalIndex, E))
3303 return this->emitGetPtrLocal(LocalIndex, E);
3306 if (!this->checkLiteralType(Inner))
3313 if (!this->VarScope->LocalsAlwaysEnabled &&
3314 !this->emitEnableLocal(*LocalIndex, E))
3317 if (!this->emitGetPtrLocal(*LocalIndex, E))
3324template <
class Emitter>
3335 if (!this->
visit(SubExpr))
3339 return this->emitPopPtr(E);
3343template <
class Emitter>
3364 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3369 if (
P.isGlobalInitialized(*GlobalIndex))
3375 return this->emitInitGlobal(*T, *GlobalIndex, E);
3387 unsigned LocalIndex;
3391 LocalIndex = *MaybeIndex;
3395 if (!this->emitGetPtrLocal(LocalIndex, E))
3399 return this->
visit(
Init) && this->emitInit(*T, E);
3403template <
class Emitter>
3416template <
class Emitter>
3420 return this->emitConst(E->
getValue(), E);
3423template <
class Emitter>
3436 for (
const Record::Field &F : R->fields()) {
3438 if (!
Init ||
Init->containsErrors())
3446 if (!this->emitInitField(*T, F.Offset, E))
3449 if (!this->emitGetPtrField(F.Offset, E))
3460template <
class Emitter>
3467 return this->emitGetPtrGlobal(StringIndex, E);
3473template <
class Emitter>
3478 return this->emitInvalid(E);
3481template <
class Emitter>
3507 if (PointeeToT && PointeeFromT) {
3526 bool Fatal = (ToT != FromT);
3533template <
class Emitter>
3536 if (!
Ctx.getLangOpts().CPlusPlus20) {
3544template <
class Emitter>
3550 return this->emitConstBool(E->
getValue(), E);
3553template <
class Emitter>
3558 if (T->isRecordType()) {
3572 if (!this->emitGetPtrLocal(*LocalIndex, E))
3580 T->getAsCXXRecordDecl()))
3590 if (!this->visitZeroRecordInitializer(R, E))
3602 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3604 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3605 if (!this->emitCheckFunctionDecl(Ctor, E))
3616 assert(
Func->hasThisPointer());
3617 assert(!
Func->hasRVO());
3621 if (!this->emitDupPtr(E))
3625 for (
const auto *Arg : E->
arguments()) {
3626 if (!this->
visit(Arg))
3630 if (
Func->isVariadic()) {
3631 uint32_t VarArgSize = 0;
3632 unsigned NumParams =
Func->getNumWrittenParams();
3633 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3637 if (!this->emitCallVar(
Func, VarArgSize, E))
3640 if (!this->emitCall(
Func, 0, E)) {
3645 (void)this->emitPopPtr(E);
3651 return this->emitPopPtr(E);
3655 if (T->isArrayType()) {
3660 if (!this->emitDupPtr(E))
3664 initArrayDimension = [&](
QualType T) ->
bool {
3665 if (!T->isArrayType()) {
3667 for (
const auto *Arg : E->
arguments()) {
3668 if (!this->
visit(Arg))
3672 return this->emitCall(
Func, 0, E);
3676 Ctx.getASTContext().getAsConstantArrayType(T);
3681 for (
size_t I = 0; I != NumElems; ++I) {
3682 if (!this->emitConstUint64(I, E))
3684 if (!this->emitArrayElemPtrUint64(E))
3686 if (!initArrayDimension(ElemTy))
3689 return this->emitPopPtr(E);
3692 return initArrayDimension(E->
getType());
3698template <
class Emitter>
3708 assert(Val.
isInt());
3710 return this->emitConst(I, E);
3717 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3718 return this->
visit(LValueExpr);
3733 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3737 const APValue &
V = UGCD->getValue();
3738 for (
unsigned I = 0, N = R->getNumFields(); I != N; ++I) {
3739 const Record::Field *F = R->getField(I);
3740 const APValue &FieldValue =
V.getStructField(I);
3746 if (!this->emitInitField(FieldT, F->Offset, E))
3754template <
class Emitter>
3760 for (
unsigned I = 0; I != N; ++I) {
3767 if (!this->
discard(ArrayIndexExpr))
3772 if (!this->
visit(ArrayIndexExpr))
3776 if (!this->emitCast(IndexT,
PT_Sint64, E))
3786 return this->emitOffsetOf(T, E, E);
3789template <
class Emitter>
3798 return this->visitZeroInitializer(*T, Ty, E);
3805 if (!this->emitGetPtrLocal(*LocalIndex, E))
3810 QualType ElemQT = CT->getElementType();
3813 for (
unsigned I = 0; I != 2; ++I) {
3814 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3816 if (!this->emitInitElem(ElemT, I, E))
3828 if (!this->emitGetPtrLocal(*LocalIndex, E))
3833 QualType ElemQT = VT->getElementType();
3836 for (
unsigned I = 0, N = VT->getNumElements(); I != N; ++I) {
3837 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3839 if (!this->emitInitElem(ElemT, I, E))
3848template <
class Emitter>
3853template <
class Emitter>
3859template <
class Emitter>
3864template <
class Emitter>
3869 return this->emitConst(E->
getValue(), E);
3872template <
class Emitter>
3877 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3891 unsigned ParamIndex = 0;
3895 if (!this->emitGetParam(PT, ParamIndex, E))
3900 return this->emitCall(F, 0, E);
3905template <
class Emitter>
3913 const Expr *PlacementDest =
nullptr;
3914 bool IsNoThrow =
false;
3919 if (PlacementArgs != 0) {
3928 if (PlacementArgs == 1) {
3936 if (!this->emitInvalidNewDeleteExpr(E, E))
3941 if (OperatorNew->isReservedGlobalPlacementOperator())
3942 PlacementDest = Arg1;
3946 return this->emitInvalid(E);
3948 }
else if (!OperatorNew
3949 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3950 return this->emitInvalidNewDeleteExpr(E, E);
3953 if (!PlacementDest) {
3958 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3961 Desc =
P.createDescriptor(
3964 false,
false,
false,
3970 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3974 const Expr *Stripped = *ArraySizeExpr;
3975 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3976 Stripped = ICE->getSubExpr())
3977 if (ICE->getCastKind() != CK_NoOp &&
3978 ICE->getCastKind() != CK_IntegralCast)
3986 if (!this->
visit(Stripped))
3988 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
3991 if (PlacementDest) {
3992 if (!this->
visit(PlacementDest))
3994 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3996 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
3999 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4004 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
4008 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
4015 size_t StaticInitElems = 0;
4016 const Expr *DynamicInit =
nullptr;
4020 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
4021 StaticInitElems = CAT->getZExtSize();
4026 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init)) {
4027 if (ILE->hasArrayFiller())
4028 DynamicInit = ILE->getArrayFiller();
4047 const Function *CtorFunc =
nullptr;
4048 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
4052 }
else if (!DynamicInit && !ElemT)
4055 LabelTy EndLabel = this->getLabel();
4056 LabelTy StartLabel = this->getLabel();
4061 if (!this->emitDupPtr(E))
4063 if (!this->emitNullPtr(0,
nullptr, E))
4065 if (!this->emitEQPtr(E))
4067 if (!this->jumpTrue(EndLabel, E))
4074 if (!this->emitConst(StaticInitElems,
SizeT, E))
4076 if (!this->emitSetLocal(
SizeT, Iter, E))
4079 this->fallthrough(StartLabel);
4080 this->emitLabel(StartLabel);
4082 if (!this->emitGetLocal(
SizeT, Iter, E))
4084 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4086 if (!this->emitLT(
SizeT, E))
4088 if (!this->jumpFalse(EndLabel, E))
4092 if (!this->emitGetLocal(
SizeT, Iter, E))
4094 if (!this->emitArrayElemPtr(
SizeT, E))
4097 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
4102 if (!this->visitZeroInitializer(InitT, ElemType, E))
4104 if (!this->emitStorePop(InitT, E))
4106 }
else if (DynamicInit) {
4108 if (!this->
visit(DynamicInit))
4110 if (!this->emitStorePop(*InitT, E))
4117 if (!this->visitZeroInitializer(
4121 if (!this->emitStorePop(*ElemT, E))
4125 if (!this->emitCall(CtorFunc, 0, E))
4130 if (!this->emitGetPtrLocal(Iter, E))
4132 if (!this->emitIncPop(
SizeT,
false, E))
4135 if (!this->jump(StartLabel, E))
4138 this->fallthrough(EndLabel);
4139 this->emitLabel(EndLabel);
4143 if (PlacementDest) {
4144 if (!this->
visit(PlacementDest))
4146 if (!this->emitCheckNewTypeMismatch(E, E))
4151 if (!this->emitAlloc(Desc, E))
4160 if (!this->emitInit(*ElemT, E))
4171 return this->emitPopPtr(E);
4176template <
class Emitter>
4182 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
4183 return this->emitInvalidNewDeleteExpr(E, E);
4192template <
class Emitter>
4198 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
4203 return this->emitGetFnPtr(
Func, E);
4206template <
class Emitter>
4210 auto canonType = [](
const Type *T) {
4211 return T->getCanonicalTypeUnqualified().getTypePtr();
4219 return this->emitGetTypeid(
4223 return this->emitGetTypeid(
4232 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
4238 if (!this->emitGetTypeidPtr(TypeInfoType, E))
4241 return this->emitPopPtr(E);
4245template <
class Emitter>
4249 return this->emitDummyPtr(E, E);
4250 return this->emitError(E);
4253template <
class Emitter>
4256 return this->emitDummyPtr(E, E);
4257 return this->emitError(E);
4260template <
class Emitter>
4262 assert(
Ctx.getLangOpts().CPlusPlus);
4263 return this->emitConstBool(E->
getValue(), E);
4266template <
class Emitter>
4278 return this->emitDummyPtr(GuidDecl, E);
4283 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
4292 assert(
V.isStruct());
4293 assert(
V.getStructNumBases() == 0);
4297 return this->emitFinishInit(E);
4300template <
class Emitter>
4310template <
class Emitter>
4319template <
class Emitter>
4325template <
class Emitter>
4329 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
4333 if (OVE->isUnique())
4349template <
class Emitter>
4354template <
class Emitter>
4356 return this->emitError(E);
4359template <
class Emitter>
4365 return this->emitDummyPtr(E, E);
4368template <
class Emitter>
4372 QualType ElemType = VT->getElementType();
4376 PrimType SrcElemT = classifyVectorElementType(SrcType);
4378 unsigned SrcOffset =
4380 if (!this->
visit(Src))
4382 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4385 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4386 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4388 if (!this->emitArrayElemPop(SrcElemT, I, E))
4392 if (SrcElemT != ElemT) {
4393 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4395 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4396 const auto *TargetSemantics = &
Ctx.getFloatSemantics(ElemType);
4400 if (!this->emitInitElem(ElemT, I, E))
4407template <
class Emitter>
4411 return this->emitInvalid(E);
4420 assert(NumOutputElems > 0);
4426 if (!this->emitGetPtrLocal(*LocalIndex, E))
4431 unsigned VectorOffsets[2];
4432 for (
unsigned I = 0; I != 2; ++I) {
4435 if (!this->
visit(Vecs[I]))
4437 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4440 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4442 assert(ShuffleIndex >= -1);
4443 if (ShuffleIndex == -1)
4444 return this->emitInvalidShuffleVectorIndex(I, E);
4446 assert(ShuffleIndex < (NumInputElems * 2));
4447 if (!this->emitGetLocal(
PT_Ptr,
4448 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4450 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4451 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4454 if (!this->emitInitElem(ElemT, I, E))
4459 return this->emitPopPtr(E);
4464template <
class Emitter>
4469 Base->getType()->isVectorType() ||
4475 if (Indices.size() == 1) {
4480 if (!this->emitConstUint32(Indices[0], E))
4482 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4492 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4500 if (!this->emitGetPtrLocal(*ResultIndex, E))
4508 uint32_t DstIndex = 0;
4509 for (uint32_t I : Indices) {
4510 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4512 if (!this->emitArrayElemPop(ElemT, I, E))
4514 if (!this->emitInitElem(ElemT, DstIndex, E))
4524template <
class Emitter>
4528 return this->
discard(SubExpr) && this->emitInvalid(E);
4534 return this->emitDummyPtr(E, E);
4537template <
class Emitter>
4542 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4547 if (!this->
visit(SubExpr))
4549 if (!this->emitConstUint8(0, E))
4551 if (!this->emitArrayElemPtrPopUint8(E))
4553 if (!this->emitInitFieldPtr(R->getField(0u)->Offset, E))
4558 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4560 return this->emitInitField(SecondFieldT, R->getField(1u)->Offset, E);
4562 assert(SecondFieldT ==
PT_Ptr);
4564 if (!this->emitGetFieldPtr(R->getField(0u)->Offset, E))
4566 if (!this->emitExpandPtr(E))
4570 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4572 return this->emitInitFieldPtr(R->getField(1u)->Offset, E);
4575template <
class Emitter>
4590 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4592 return this->emitUnsupported(E);
4601 return this->
Visit(E);
4608 return this->
Visit(E);
4612 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4615 if (
const auto *CE = dyn_cast<CastExpr>(E);
4617 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4624 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4627 if (
const auto *CE = dyn_cast<CastExpr>(E);
4628 CE && (CE->getCastKind() == CK_DerivedToBase ||
4629 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4630 CE->getCastKind() == CK_NoOp))
4650 if (!this->emitGetPtrLocal(*LocalIndex, E))
4660 return this->
Visit(E);
4663template <
class Emitter>
4669 return this->
Visit(E) && this->emitFinishInit(E);
4672template <
class Emitter>
4678 return this->
Visit(E) && this->emitFinishInitPop(E);
4684 return this->
Visit(E);
4695 if (!this->
visit(E))
4697 return this->emitComplexBoolCast(E);
4702 if (!this->
visit(E))
4710 return this->emitIsNonNullPtr(E);
4714 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4717 return this->emitCast(*T,
PT_Bool, E);
4720template <
class Emitter>
4724 QT = AT->getValueType();
4728 return this->emitZeroBool(E);
4730 return this->emitZeroSint8(E);
4732 return this->emitZeroUint8(E);
4734 return this->emitZeroSint16(E);
4736 return this->emitZeroUint16(E);
4738 return this->emitZeroSint32(E);
4740 return this->emitZeroUint32(E);
4742 return this->emitZeroSint64(E);
4744 return this->emitZeroUint64(E);
4746 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4748 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4750 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4753 return this->emitNullMemberPtr(0,
nullptr, E);
4755 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4756 return this->emitFloat(F, E);
4759 auto Sem = Ctx.getASTContext().getFixedPointSemantics(E->
getType());
4763 llvm_unreachable(
"unknown primitive type");
4766template <
class Emitter>
4767bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4772 for (
const Record::Field &Field :
R->fields()) {
4773 if (Field.isUnnamedBitField())
4780 if (!this->visitZeroInitializer(T, QT, E))
4783 if (!this->emitInitFieldActivate(T, Field.Offset, E))
4787 if (!this->emitInitField(T, Field.Offset, E))
4792 if (!this->emitGetPtrField(Field.Offset, E))
4798 for (uint32_t I = 0, N = D->
getNumElems(); I != N; ++I) {
4799 if (!this->visitZeroInitializer(T, ET, E))
4801 if (!this->emitInitElem(T, I, E))
4806 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4809 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4817 if (!this->emitFinishInitActivatePop(E))
4821 if (!this->emitFinishInitPop(E))
4825 for (
const Record::Base &B :
R->bases()) {
4826 if (!this->emitGetPtrBase(B.Offset, E))
4828 if (!this->visitZeroRecordInitializer(B.R, E))
4830 if (!this->emitFinishInitPop(E))
4839template <
class Emitter>
4840bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4841 assert(T->isArrayType() || T->isAnyComplexType() || T->isVectorType());
4847 for (
size_t I = 0; I != NumElems; ++I) {
4848 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4850 if (!this->emitInitElem(*ElemT, I, E))
4856 const Record *
R = getRecord(ElemType);
4860 for (
size_t I = 0; I != NumElems; ++I) {
4861 if (!this->emitConstUint32(I, E))
4863 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4865 if (!this->visitZeroRecordInitializer(R, E))
4867 if (!this->emitPopPtr(E))
4873 for (
size_t I = 0; I != NumElems; ++I) {
4874 if (!this->emitConstUint32(I, E))
4876 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4878 if (!this->visitZeroArrayInitializer(ElemType, E))
4880 if (!this->emitPopPtr(E))
4889template <
class Emitter>
4890bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4892 if (!canClassify(E->
getType()))
4895 if (!this->visit(RHS))
4897 if (!this->visit(LHS))
4904 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4908 bool Activates = refersToUnion(LHS);
4911 if (!this->emitFlip(
PT_Ptr, RHT, E))
4914 if (DiscardResult) {
4915 if (BitField && Activates)
4916 return this->emitStoreBitFieldActivatePop(RHT, E);
4918 return this->emitStoreBitFieldPop(RHT, E);
4920 return this->emitStoreActivatePop(RHT, E);
4922 return this->emitStorePop(RHT, E);
4925 auto maybeLoad = [&](
bool Result) ->
bool {
4931 return this->emitLoadPop(RHT, E);
4935 if (BitField && Activates)
4936 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4938 return maybeLoad(this->emitStoreBitField(RHT, E));
4940 return maybeLoad(this->emitStoreActivate(RHT, E));
4942 return maybeLoad(this->emitStore(RHT, E));
4945template <
class Emitter>
4946template <
typename T>
4950 return this->emitConstSint8(
Value, E);
4952 return this->emitConstUint8(
Value, E);
4954 return this->emitConstSint16(
Value, E);
4956 return this->emitConstUint16(
Value, E);
4958 return this->emitConstSint32(
Value, E);
4960 return this->emitConstUint32(
Value, E);
4962 return this->emitConstSint64(
Value, E);
4964 return this->emitConstUint64(
Value, E);
4966 return this->emitConstBool(
Value, E);
4973 llvm_unreachable(
"Invalid integral type");
4976 llvm_unreachable(
"unknown primitive type");
4979template <
class Emitter>
4980template <
typename T>
4981bool Compiler<Emitter>::emitConst(T
Value,
const Expr *E) {
4982 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
4985template <
class Emitter>
4989 return this->emitConstIntAPS(
Value, E);
4991 return this->emitConstIntAP(
Value, E);
4993 if (
Value.isSigned())
4994 return this->emitConst(
Value.getSExtValue(), Ty, E);
4995 return this->emitConst(
Value.getZExtValue(), Ty, E);
4998template <
class Emitter>
5002 return this->emitConstIntAPS(
Value, E);
5004 return this->emitConstIntAP(
Value, E);
5007 return this->emitConst(
Value.getSExtValue(), Ty, E);
5008 return this->emitConst(
Value.getZExtValue(), Ty, E);
5011template <
class Emitter>
5012bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
5013 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5016template <
class Emitter>
5029 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
5030 Locals.insert({VD, Local});
5031 VarScope->addForScopeKind(Local, SC);
5032 return Local.Offset;
5035template <
class Emitter>
5040 bool IsTemporary =
false;
5041 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
5044 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
5045 Init = VarD->getInit();
5047 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
5058 return std::nullopt;
5063 Locals.insert({Key, Local});
5064 VarScope->addForScopeKind(Local, SC);
5065 return Local.Offset;
5068template <
class Emitter>
5078 return std::nullopt;
5088 return Local.Offset;
5091template <
class Emitter>
5093 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
5094 return PT->getPointeeType()->getAsCanonical<RecordType>();
5100 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
5104template <
class Emitter>
5106 return P.getOrCreateRecord(RD);
5109template <
class Emitter>
5111 return Ctx.getOrCreateFunction(FD);
5114template <
class Emitter>
5118 auto maybeDestroyLocals = [&]() ->
bool {
5119 if (DestroyToplevelScope)
5120 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
5121 return this->emitCheckAllocations(E);
5128 return this->emitRetVoid(E) && maybeDestroyLocals();
5136 return this->emitRet(*T, E) && maybeDestroyLocals();
5144 if (!this->emitGetPtrLocal(*LocalOffset, E))
5152 return this->emitRetValue(E) && maybeDestroyLocals();
5155 return maybeDestroyLocals() &&
false;
5158template <
class Emitter>
5160 bool DestroyToplevelScope) {
5164 return this->
visitExpr(E, DestroyToplevelScope);
5167template <
class Emitter>
5179 if (
auto GlobalIndex =
P.getGlobal(VD)) {
5180 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5194template <
class Emitter>
5196 bool ConstantContext) {
5199 if (!ConstantContext) {
5213 auto GlobalIndex =
P.getGlobal(VD);
5214 assert(GlobalIndex);
5216 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
5219 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
5223 auto Local =
Locals.find(VD);
5224 assert(Local !=
Locals.end());
5226 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
5229 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
5239 auto GlobalIndex =
P.getGlobal(VD);
5240 assert(GlobalIndex);
5241 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5250 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
5253template <
class Emitter>
5264 if (!this->isActive())
5269 if (
Init &&
Init->isValueDependent())
5273 auto checkDecl = [&]() ->
bool {
5275 return !NeedsOp || this->emitCheckDecl(VD, VD);
5282 if (!
P.getGlobal(*GlobalIndex)->isInitialized())
5285 if (
P.isGlobalInitialized(*GlobalIndex))
5289 }
else if ((GlobalIndex =
P.createGlobal(VD,
Init))) {
5303 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
5306 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
5309 if (!this->emitStartInit(
Init))
5315 if (!this->emitEndInit(
Init))
5318 return this->emitFinishInitGlobal(
Init);
5328 if (!
Init ||
Init->getType()->isVoidType())
5337 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
5349 if (!this->emitCheckRefInit(
Init))
5353 return this->emitSetLocal(*VarT, Offset, VD);
5361 if (!this->emitGetPtrLocal(*Offset,
Init))
5369template <
class Emitter>
5374 return this->emitConst(Val.
getInt(), ValType, E);
5377 return this->emitFloat(F, E);
5382 if (!this->emitGetMemberPtr(MemberDecl, E))
5388 if (!this->emitCopyMemberPtrPath(PathEntry, IsDerived, E))
5394 return this->emitNullMemberPtr(0,
nullptr, E);
5399 return this->emitNull(ValType, 0,
nullptr, E);
5404 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
5405 return this->
visit(BaseExpr);
5410 QualType EntryType = VD->getType();
5411 for (
auto &Entry : Path) {
5413 uint64_t Index = Entry.getAsArrayIndex();
5416 if (!this->emitConst(Index,
PT_Uint64, E))
5418 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
5420 EntryType = ElemType;
5430 const Decl *BaseOrMember = Entry.getAsBaseOrMember().getPointer();
5431 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
5433 if (!this->emitGetPtrFieldPop(EntryOffset, E))
5435 EntryType = FD->getType();
5439 if (!this->emitGetPtrBasePop(BaseOffset,
false, E))
5441 EntryType =
Ctx.getASTContext().getCanonicalTagType(
Base);
5453template <
class Emitter>
5461 const Record::Field *RF = R->getField(I);
5462 QualType FieldType = RF->Decl->getType();
5468 if (!this->emitInitField(*PT, RF->Offset, E))
5471 if (!this->emitGetPtrField(RF->Offset, E))
5475 if (!this->emitFinishInitPop(E))
5483 const Record::Base *RB = R->getBase(I);
5484 QualType BaseType =
Ctx.getASTContext().getCanonicalTagType(RB->Decl);
5486 if (!this->emitGetPtrBase(RB->Offset, E))
5490 if (!this->emitFinishInitPop(E))
5503 const Record::Field *RF = R->getField(UnionField);
5504 QualType FieldType = RF->Decl->getType();
5509 if (RF->isBitField())
5510 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
5512 return this->emitInitFieldActivate(*PT, RF->Offset, E);
5515 if (!this->emitGetPtrField(RF->Offset, E))
5517 if (!this->emitActivate(E))
5521 return this->emitPopPtr(E);
5524 const auto *ArrType = T->getAsArrayTypeUnsafe();
5525 QualType ElemType = ArrType->getElementType();
5526 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5531 if (!this->emitInitElem(*ElemT, A, E))
5534 if (!this->emitConstUint32(A, E))
5536 if (!this->emitArrayElemPtrUint32(E))
5540 if (!this->emitPopPtr(E))
5551template <
class Emitter>
5553 unsigned BuiltinID) {
5554 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5559 return this->emitConst(0, E);
5562 if (!this->emitStartSpeculation(E))
5564 LabelTy EndLabel = this->getLabel();
5565 if (!this->speculate(E, EndLabel))
5567 if (!this->emitEndSpeculation(E))
5569 this->fallthrough(EndLabel);
5577 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5578 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5579 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5580 BuiltinID == Builtin::BI__builtin_function_start) {
5583 return this->emitDummyPtr(E, E);
5594 if (!this->emitGetPtrLocal(*LocalIndex, E))
5599 switch (BuiltinID) {
5600 case Builtin::BI__builtin_object_size:
5601 case Builtin::BI__builtin_dynamic_object_size: {
5605 if (!this->
visit(Arg0))
5616 case Builtin::BI__assume:
5617 case Builtin::BI__builtin_assume:
5620 case Builtin::BI__atomic_is_lock_free:
5621 case Builtin::BI__atomic_always_lock_free: {
5632 for (
const auto *Arg : E->
arguments()) {
5633 if (!this->
visit(Arg))
5639 if (!this->emitCallBI(E, BuiltinID, E))
5648template <
class Emitter>
5667 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5668 DD && DD->isTrivial()) {
5670 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5672 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5673 this->emitPopPtr(E);
5681 bool HasRVO = !
ReturnType->isVoidType() && !T;
5689 if (!this->emitGetPtrLocal(*LocalIndex, E))
5697 if (!this->emitGetPtrLocal(*LocalIndex, E))
5701 if (!this->emitDupPtr(E))
5708 bool IsAssignmentOperatorCall =
false;
5709 bool ActivateLHS =
false;
5710 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5711 OCE && OCE->isAssignmentOp()) {
5715 assert(Args.size() == 2);
5716 const CXXRecordDecl *LHSRecord = Args[0]->getType()->getAsCXXRecordDecl();
5718 IsAssignmentOperatorCall =
true;
5719 std::reverse(Args.begin(), Args.end());
5725 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5726 MD && MD->isStatic()) {
5730 Args.erase(Args.begin());
5734 bool Devirtualized =
false;
5737 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5745 if (!this->
visit(Callee))
5747 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5749 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5751 if (!this->emitGetMemberPtrBase(E))
5754 const auto *InstancePtr = MC->getImplicitObjectArgument();
5761 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5762 Devirtualized =
true;
5763 if (!this->
visit(Stripped))
5766 if (!this->
visit(InstancePtr))
5770 if (!this->
visit(InstancePtr))
5774 }
else if (
const auto *PD =
5775 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5776 if (!this->emitCheckPseudoDtor(E))
5784 return this->emitEndLifetimePop(E);
5785 }
else if (!FuncDecl) {
5789 if (!this->
visit(Callee))
5791 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5800 if (IsAssignmentOperatorCall) {
5801 assert(Args.size() == 2);
5804 if (!this->emitFlip(Arg2T, Arg1T, E))
5818 assert(HasRVO ==
Func->hasRVO());
5820 bool HasQualifier =
false;
5821 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5822 HasQualifier = ME->hasQualifier();
5824 bool IsVirtual =
false;
5825 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5826 IsVirtual = !Devirtualized && MD->isVirtual();
5831 if (IsVirtual && !HasQualifier) {
5832 uint32_t VarArgSize = 0;
5833 unsigned NumParams =
5835 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5838 if (!this->emitCallVirt(
Func, VarArgSize, E))
5840 }
else if (
Func->isVariadic()) {
5841 uint32_t VarArgSize = 0;
5842 unsigned NumParams =
5844 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5846 if (!this->emitCallVar(
Func, VarArgSize, E))
5849 if (!this->emitCall(
Func, 0, E))
5858 uint32_t ArgSize = 0;
5859 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5865 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5867 if (!this->emitGetMemberPtrDecl(E))
5870 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5873 if (!this->emitCallPtr(ArgSize, E, E))
5884template <
class Emitter>
5891template <
class Emitter>
5898template <
class Emitter>
5903 return this->emitConstBool(E->
getValue(), E);
5906template <
class Emitter>
5912 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5913 return this->emitNullPtr(Val,
nullptr, E);
5916template <
class Emitter>
5924 return this->emitZero(T, E);
5927template <
class Emitter>
5932 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
5933 if (this->LambdaThisCapture.Offset > 0) {
5934 if (this->LambdaThisCapture.IsPtr)
5935 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
5936 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
5945 return this->emitThis(E);
5960 unsigned StartIndex = 0;
5961 unsigned EndIndex = 0;
5963 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
5965 EndIndex = StartIndex;
5972 for (; StartIndex > 0; --StartIndex) {
5982 if (StartIndex == 0 && EndIndex == 0)
5985 assert(StartIndex < EndIndex);
5988 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
6000 case Stmt::CompoundStmtClass:
6002 case Stmt::DeclStmtClass:
6004 case Stmt::ReturnStmtClass:
6006 case Stmt::IfStmtClass:
6008 case Stmt::WhileStmtClass:
6010 case Stmt::DoStmtClass:
6012 case Stmt::ForStmtClass:
6014 case Stmt::CXXForRangeStmtClass:
6016 case Stmt::BreakStmtClass:
6018 case Stmt::ContinueStmtClass:
6020 case Stmt::SwitchStmtClass:
6022 case Stmt::CaseStmtClass:
6024 case Stmt::DefaultStmtClass:
6026 case Stmt::AttributedStmtClass:
6028 case Stmt::CXXTryStmtClass:
6030 case Stmt::NullStmtClass:
6033 case Stmt::GCCAsmStmtClass:
6034 case Stmt::MSAsmStmtClass:
6035 case Stmt::GotoStmtClass:
6036 return this->emitInvalid(S);
6037 case Stmt::LabelStmtClass:
6040 if (
const auto *E = dyn_cast<Expr>(S))
6047template <
class Emitter>
6050 for (
const auto *InnerStmt : S->
body())
6053 return Scope.destroyLocals();
6056template <
class Emitter>
6057bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
6058 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
6059 for (
auto *BD : DD->flat_bindings())
6060 if (
auto *KD = BD->getHoldingVar();
6061 KD && !this->visitVarDecl(KD, KD->getInit()))
6075template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
6077 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
6078 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
6085 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
6090 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
6091 ICE && (ICE->getCastKind() == CK_NoOp ||
6092 ICE->getCastKind() == CK_DerivedToBase ||
6093 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
6094 E = ICE->getSubExpr();
6098 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
6099 const auto *ThisRecord =
6100 This->getType()->getPointeeType()->getAsRecordDecl();
6101 if (!ThisRecord->isUnion())
6104 if (
const auto *Ctor =
6105 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
6106 return Ctor->getParent() == ThisRecord;
6115template <
class Emitter>
6117 bool EvaluateConditionDecl) {
6118 for (
const auto *D : DS->
decls()) {
6123 const auto *VD = dyn_cast<VarDecl>(D);
6130 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
6137template <
class Emitter>
6140 return this->emitUnsupported(RS);
6146 if (!this->
visit(RE))
6152 if (RE->getType()->isVoidType()) {
6153 if (!this->
visit(RE))
6156 if (RE->containsErrors())
6161 if (!this->emitRVOPtr(RE))
6167 return this->emitRetVoid(RS);
6173 return this->emitRetVoid(RS);
6179 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
6186 if (
auto *CondInit = IS->
getInit()) {
6202 return visitChildStmt(IS->
getThen());
6204 return visitChildStmt(Else);
6210 if (!this->emitIsConstantContext(IS))
6213 if (!this->emitIsConstantContext(IS))
6215 if (!this->emitInv(IS))
6229 LabelTy LabelElse = this->getLabel();
6230 LabelTy LabelEnd = this->getLabel();
6231 if (!this->jumpFalse(LabelElse, IS))
6233 if (!visitChildStmt(IS->
getThen()))
6235 if (!this->jump(LabelEnd, IS))
6237 this->emitLabel(LabelElse);
6238 if (!visitChildStmt(Else))
6240 this->emitLabel(LabelEnd);
6242 LabelTy LabelEnd = this->getLabel();
6243 if (!this->jumpFalse(LabelEnd, IS))
6245 if (!visitChildStmt(IS->
getThen()))
6247 this->emitLabel(LabelEnd);
6256template <
class Emitter>
6261 LabelTy CondLabel = this->getLabel();
6262 LabelTy EndLabel = this->getLabel();
6266 this->fallthrough(CondLabel);
6267 this->emitLabel(CondLabel);
6283 if (!this->jumpFalse(EndLabel, S))
6293 if (!this->jump(CondLabel, S))
6295 this->fallthrough(EndLabel);
6296 this->emitLabel(EndLabel);
6305 LabelTy StartLabel = this->getLabel();
6306 LabelTy EndLabel = this->getLabel();
6307 LabelTy CondLabel = this->getLabel();
6311 this->fallthrough(StartLabel);
6312 this->emitLabel(StartLabel);
6318 this->fallthrough(CondLabel);
6319 this->emitLabel(CondLabel);
6326 if (!this->jumpTrue(StartLabel, S))
6329 this->fallthrough(EndLabel);
6330 this->emitLabel(EndLabel);
6334template <
class Emitter>
6342 LabelTy EndLabel = this->getLabel();
6343 LabelTy CondLabel = this->getLabel();
6344 LabelTy IncLabel = this->getLabel();
6351 this->fallthrough(CondLabel);
6352 this->emitLabel(CondLabel);
6364 if (!this->jumpFalse(EndLabel, S))
6373 this->fallthrough(IncLabel);
6374 this->emitLabel(IncLabel);
6380 if (!this->jump(CondLabel, S))
6384 this->emitLabel(EndLabel);
6390template <
class Emitter>
6400 LabelTy EndLabel = this->getLabel();
6401 LabelTy CondLabel = this->getLabel();
6402 LabelTy IncLabel = this->getLabel();
6417 this->fallthrough(CondLabel);
6418 this->emitLabel(CondLabel);
6421 if (!this->jumpFalse(EndLabel, S))
6432 this->fallthrough(IncLabel);
6433 this->emitLabel(IncLabel);
6438 if (!this->jump(CondLabel, S))
6441 this->fallthrough(EndLabel);
6442 this->emitLabel(EndLabel);
6446template <
class Emitter>
6457 if (LI.BreakLabel) {
6458 TargetLabel = *LI.BreakLabel;
6459 BreakScope = LI.BreakOrContinueScope;
6465 if (LI.Name == TargetLoop) {
6466 TargetLabel = *LI.BreakLabel;
6467 BreakScope = LI.BreakOrContinueScope;
6478 C =
C->getParent()) {
6479 if (!
C->destroyLocals())
6483 return this->jump(*TargetLabel, S);
6486template <
class Emitter>
6497 if (LI.ContinueLabel) {
6498 TargetLabel = *LI.ContinueLabel;
6499 ContinueScope = LI.BreakOrContinueScope;
6505 if (LI.Name == TargetLoop) {
6506 TargetLabel = *LI.ContinueLabel;
6507 ContinueScope = LI.BreakOrContinueScope;
6512 assert(TargetLabel);
6515 C =
C->getParent()) {
6516 if (!
C->destroyLocals())
6520 return this->jump(*TargetLabel, S);
6523template <
class Emitter>
6526 if (
Cond->containsErrors())
6532 LabelTy EndLabel = this->getLabel();
6537 if (
const auto *CondInit = S->
getInit())
6548 if (!this->emitSetLocal(CondT, CondVar, S))
6558 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6561 if (CS->caseStmtIsGNURange()) {
6562 LabelTy EndOfRangeCheck = this->getLabel();
6563 const Expr *Low = CS->getLHS();
6564 const Expr *High = CS->getRHS();
6568 if (!this->emitGetLocal(CondT, CondVar, CS))
6570 if (!this->
visit(Low))
6573 if (!this->emitGE(
LT, S))
6575 if (!this->jumpFalse(EndOfRangeCheck, S))
6578 if (!this->emitGetLocal(CondT, CondVar, CS))
6580 if (!this->
visit(High))
6583 if (!this->emitLE(HT, S))
6587 this->emitLabel(EndOfRangeCheck);
6592 if (
Value->isValueDependent())
6597 if (!this->emitGetLocal(CondT, CondVar, CS))
6603 if (!this->emitEQ(ValueT, S))
6608 assert(!DefaultLabel);
6609 DefaultLabel = this->getLabel();
6616 if (!this->jump(*DefaultLabel, S))
6619 if (!this->jump(EndLabel, S))
6627 this->fallthrough(EndLabel);
6628 this->emitLabel(EndLabel);
6633template <
class Emitter>
6640template <
class Emitter>
6647 if (LI.DefaultLabel) {
6648 DefaultLabel = *LI.DefaultLabel;
6653 this->emitLabel(DefaultLabel);
6657template <
class Emitter>
6664 if (IsMSVCConstexprAttr && !this->emitPushMSVCCE(S))
6667 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6668 !this->Ctx.getLangOpts().MSVCCompat) {
6670 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6676 const Expr *Assumption = AA->getAssumption();
6687 if (!this->emitAssume(Assumption))
6696 if (IsMSVCConstexprAttr)
6697 return this->emitPopMSVCCE(S);
6701template <
class Emitter>
6707template <
class Emitter>
6708bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6715 assert(ClosureClass->
captures().empty());
6719 "A generic lambda's static-invoker function must be a "
6720 "template specialization");
6724 void *InsertPos =
nullptr;
6725 const FunctionDecl *CorrespondingCallOpSpecialization =
6727 assert(CorrespondingCallOpSpecialization);
6728 LambdaCallOp = CorrespondingCallOpSpecialization;
6732 assert(ClosureClass->
captures().empty());
6733 const Function *
Func = this->getFunction(LambdaCallOp);
6736 assert(
Func->hasThisPointer());
6739 if (
Func->hasRVO()) {
6740 if (!this->emitRVOPtr(MD))
6748 if (!this->emitNullPtr(0,
nullptr, MD))
6753 auto It = this->Params.find(PVD);
6754 assert(It != this->Params.end());
6758 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6759 if (!this->emitGetParam(ParamType, It->second.Index, MD))
6763 if (!this->emitCall(
Func, 0, LambdaCallOp))
6768 return this->emitRet(*ReturnType, MD);
6771 return this->emitRetVoid(MD);
6774template <
class Emitter>
6775bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6776 if (Ctx.getLangOpts().CPlusPlus23)
6786 const Expr *InitExpr =
Init->getInit();
6788 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6792 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6802template <
class Emitter>
6804 assert(!ReturnType);
6807 if (!this->emitStartThisLifetime1(Ctor))
6810 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6811 const Expr *InitExpr,
6814 if (InitExpr->getType().isNull())
6818 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6821 if (!this->visit(InitExpr))
6824 if (F->isBitField())
6825 return this->emitInitThisBitField(*T, FieldOffset, F->bitWidth(),
6827 return this->emitInitThisField(*T, FieldOffset, InitExpr);
6832 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6835 if (
Activate && !this->emitActivate(InitExpr))
6838 return this->visitInitializerPop(InitExpr);
6842 const Record *
R = this->getRecord(RD);
6845 bool IsUnion =
R->isUnion();
6855 if (!this->emitThis(Ctor))
6858 if (!this->emitGetParam(
PT_Ptr, 0, Ctor))
6861 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6862 this->emitRetVoid(Ctor);
6865 unsigned FieldInits = 0;
6867 for (
const auto *
Init : Ctor->
inits()) {
6871 const Expr *InitExpr =
Init->getInit();
6873 const Record::Field *F =
R->getField(
Member);
6877 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6881 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6884 if (
Init->isBaseVirtual()) {
6885 assert(
R->getVirtualBase(BaseDecl));
6886 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6892 const Record::Base *B =
R->getBase(BaseDecl);
6894 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6898 if (IsUnion && !this->emitActivate(InitExpr))
6901 if (!this->visitInitializerPop(InitExpr))
6906 unsigned ChainSize = IFD->getChainingSize();
6907 assert(ChainSize >= 2);
6909 unsigned NestedFieldOffset = 0;
6910 const Record::Field *NestedField =
nullptr;
6911 for (
unsigned I = 0; I != ChainSize; ++I) {
6913 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6914 assert(FieldRecord);
6916 NestedField = FieldRecord->
getField(FD);
6917 assert(NestedField);
6918 IsUnion = IsUnion || FieldRecord->
isUnion();
6920 NestedFieldOffset += NestedField->Offset;
6923 if (I != ChainSize - 1)
6926 assert(NestedField);
6929 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
6934 unsigned InitFieldOffset = 0;
6935 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
6937 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6938 assert(FieldRecord);
6939 NestedField = FieldRecord->
getField(FD);
6940 InitFieldOffset += NestedField->Offset;
6941 assert(NestedField);
6942 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
6944 if (!this->emitFinishInitPop(InitExpr))
6948 InitStack.pop_back_n(ChainSize - 1);
6951 assert(
Init->isDelegatingInitializer());
6952 if (!this->emitThis(InitExpr))
6954 if (!this->visitInitializerPop(
Init->getInit()))
6958 if (!
Scope.destroyLocals())
6962 if (FieldInits !=
R->getNumFields()) {
6963 assert(FieldInits < R->getNumFields());
6965 if (!this->emitStartThisLifetime(Ctor))
6972 if (
const auto *CS = dyn_cast<CompoundStmt>(Body)) {
6973 if (!CS->body_empty() && !this->emitCtorCheck(
SourceInfo{}))
6980 if (!visitStmt(Body))
6987template <
class Emitter>
6990 const Record *
R = this->getRecord(RD);
6995 if (!this->visitStmt(Dtor->
getBody()))
6999 if (!this->emitThis(Dtor))
7002 if (!this->emitCheckDestruction(Dtor))
7006 if (!
R->isUnion()) {
7010 for (
const Record::Field &Field : llvm::reverse(
R->fields())) {
7014 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
7016 if (!this->emitDestructionPop(D,
SourceInfo{}))
7021 for (
const Record::Base &
Base : llvm::reverse(
R->bases())) {
7022 if (
Base.R->hasTrivialDtor())
7026 if (!this->emitRecordDestructionPop(
Base.R, {}))
7030 if (!this->emitMarkDestroyed(Dtor))
7034 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
7037template <
class Emitter>
7038bool Compiler<Emitter>::compileUnionAssignmentOperator(
7040 if (!this->emitThis(MD))
7043 if (!this->emitGetParam(
PT_Ptr, 0, MD))
7046 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
7049template <
class Emitter>
7059 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
7060 return this->compileConstructor(Ctor);
7061 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
7062 return this->compileDestructor(Dtor);
7065 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
7070 return this->compileUnionAssignmentOperator(MD);
7073 return this->emitLambdaStaticInvokerBody(MD);
7077 if (
const auto *Body = F->
getBody())
7091 return FD->getBitWidthValue();
7094template <
class Emitter>
7107 if (!
Ctx.getLangOpts().CPlusPlus14)
7108 return this->emitInvalid(E);
7110 return this->emitError(E);
7112 if (!this->
visit(SubExpr))
7116 if (!this->emitIncPtr(E))
7123 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
7124 : this->emitIncf(getFPOptions(E), E);
7136 if (!
Ctx.getLangOpts().CPlusPlus14)
7137 return this->emitInvalid(E);
7139 return this->emitError(E);
7141 if (!this->
visit(SubExpr))
7145 if (!this->emitDecPtr(E))
7152 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
7153 : this->emitDecf(getFPOptions(E), E);
7166 if (!
Ctx.getLangOpts().CPlusPlus14)
7167 return this->emitInvalid(E);
7169 return this->emitError(E);
7171 if (!this->
visit(SubExpr))
7175 if (!this->emitLoadPtr(E))
7177 if (!this->emitConstUint8(1, E))
7179 if (!this->emitAddOffsetUint8(E))
7181 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7187 return this->emitIncfPop(getFPOptions(E), E);
7197 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7198 if (!this->emitLoadFloat(E))
7200 APFloat F(TargetSemantics, 1);
7201 if (!this->emitFloat(F, E))
7204 if (!this->emitAddf(getFPOptions(E), E))
7206 if (!this->emitStoreFloat(E))
7218 return E->
isGLValue() || this->emitLoadPop(*T, E);
7221 if (!
Ctx.getLangOpts().CPlusPlus14)
7222 return this->emitInvalid(E);
7224 return this->emitError(E);
7226 if (!this->
visit(SubExpr))
7230 if (!this->emitLoadPtr(E))
7232 if (!this->emitConstUint8(1, E))
7234 if (!this->emitSubOffsetUint8(E))
7236 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7242 return this->emitDecfPop(getFPOptions(E), E);
7252 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7253 if (!this->emitLoadFloat(E))
7255 APFloat F(TargetSemantics, 1);
7256 if (!this->emitFloat(F, E))
7259 if (!this->emitSubf(getFPOptions(E), E))
7261 if (!this->emitStoreFloat(E))
7273 return E->
isGLValue() || this->emitLoadPop(*T, E);
7277 return this->emitError(E);
7280 return this->
discard(SubExpr);
7285 if (!this->emitInv(E))
7289 return this->emitCast(
PT_Bool, ET, E);
7293 return this->emitError(E);
7295 if (!this->
visit(SubExpr))
7297 return DiscardResult ? this->emitPop(*T, E) : this->emitNeg(*T, E);
7300 return this->emitError(E);
7302 if (!this->
visit(SubExpr))
7317 return this->
discard(SubExpr);
7319 if (!this->
visit(SubExpr))
7326 return this->emitNarrowPtr(E);
7331 return this->emitError(E);
7333 if (!this->
visit(SubExpr))
7335 return DiscardResult ? this->emitPop(*T, E) : this->emitComp(*T, E);
7347 : this->visitZeroInitializer(*T, SubExpr->
getType(), SubExpr);
7352 assert(
false &&
"Unhandled opcode");
7358template <
class Emitter>
7364 return this->
discard(SubExpr);
7367 auto prepareResult = [=]() ->
bool {
7372 return this->emitGetPtrLocal(*LocalIndex, E);
7379 unsigned SubExprOffset = ~0u;
7380 auto createTemp = [=, &SubExprOffset]() ->
bool {
7383 if (!this->
visit(SubExpr))
7385 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
7389 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7390 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7392 return this->emitArrayElemPop(ElemT, Index, E);
7397 if (!prepareResult())
7401 for (
unsigned I = 0; I != 2; ++I) {
7402 if (!getElem(SubExprOffset, I))
7404 if (!this->emitNeg(ElemT, E))
7406 if (!this->emitInitElem(ElemT, I, E))
7417 if (!this->
visit(SubExpr))
7419 if (!this->emitComplexBoolCast(SubExpr))
7421 if (!this->emitInv(E))
7424 return this->emitCast(
PT_Bool, ET, E);
7428 return this->emitComplexReal(SubExpr);
7431 if (!this->
visit(SubExpr))
7435 if (!this->emitConstUint8(1, E))
7437 return this->emitArrayElemPtrPopUint8(E);
7448 if (!this->emitArrayElem(ElemT, 1, E))
7450 if (!this->emitNeg(ElemT, E))
7452 if (!this->emitInitElem(ElemT, 1, E))
7460 return this->emitInvalid(E);
7466template <
class Emitter>
7472 return this->
discard(SubExpr);
7475 if (UnaryOp == UO_Extension)
7478 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
7479 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
7480 return this->emitInvalid(E);
7483 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
7490 if (!this->emitGetPtrLocal(*LocalIndex, E))
7495 unsigned SubExprOffset =
7497 if (!this->
visit(SubExpr))
7499 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
7504 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7505 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7507 return this->emitArrayElemPop(ElemT, Index, E);
7512 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7513 if (!getElem(SubExprOffset, I))
7515 if (!this->emitNeg(ElemT, E))
7517 if (!this->emitInitElem(ElemT, I, E))
7532 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7533 if (!getElem(SubExprOffset, I))
7536 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
7538 if (!this->emitInv(E))
7540 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
7542 if (!this->emitNeg(ElemT, E))
7544 if (ElemT != ResultVecElemT &&
7545 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
7547 if (!this->emitInitElem(ResultVecElemT, I, E))
7553 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7554 if (!getElem(SubExprOffset, I))
7557 if (!this->emitInv(E))
7560 if (!this->emitComp(ElemT, E))
7563 if (!this->emitInitElem(ElemT, I, E))
7568 llvm_unreachable(
"Unsupported unary operators should be handled up front");
7573template <
class Emitter>
7578 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D))
7579 return this->emitConst(ECD->getInitVal(), E);
7580 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
7582 return F && this->emitGetFnPtr(F, E);
7584 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
7589 return this->emitInitGlobal(*T, *Index, E);
7592 if (!this->emitGetPtrGlobal(*Index, E))
7596 return this->emitFinishInit(E);
7611 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
7612 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7617 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7618 if (IsReference || !It->second.IsPtr)
7619 return this->emitGetParam(
classifyPrim(E), It->second.Index, E);
7621 return this->emitGetPtrParam(It->second.Index, E);
7624 if (!
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7630 const unsigned Offset = It->second.Offset;
7633 return this->emitGetRefLocal(Offset, E);
7635 return this->emitGetPtrLocal(Offset, E);
7638 if (
auto GlobalIndex =
P.getGlobal(D)) {
7640 if (!
Ctx.getLangOpts().CPlusPlus11)
7641 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7642 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7645 return this->emitGetPtrGlobal(*GlobalIndex, E);
7649 auto revisit = [&](
const VarDecl *VD,
7650 bool IsConstexprUnknown =
true) ->
bool {
7652 IsConstexprUnknown);
7657 if (!this->emitPopCC(E))
7660 if (VarState.notCreated())
7668 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
7670 if (
auto It = this->LambdaCaptures.find(D);
7671 It != this->LambdaCaptures.end()) {
7672 auto [Offset, IsPtr] = It->second;
7675 return this->emitGetThisFieldPtr(Offset, E);
7676 return this->emitGetPtrThisField(Offset, E);
7680 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7681 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7682 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7686 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7687 return this->
visit(BD->getBinding());
7691 return this->emitDummyPtr(D, E);
7695 const auto *VD = dyn_cast<VarDecl>(D);
7697 return this->emitError(E);
7700 if (!
Ctx.getLangOpts().CPlusPlus) {
7704 return revisit(VD,
false);
7705 return this->emitDummyPtr(D, E);
7709 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7710 if (T.isConstant(
Ctx.getASTContext()))
7712 return T->isReferenceType();
7716 typeShouldBeVisited(DeclType)) {
7718 Init && !
Init->isValueDependent()) {
7724 (void)
Init->EvaluateAsInitializer(
V,
Ctx.getASTContext(), VD, Notes,
7738 bool IsConstexprUnknown = !DeclType.
isConstant(
Ctx.getASTContext()) &&
7743 return revisit(VD, IsConstexprUnknown);
7744 }
else if (
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7745 return revisit(VD,
true);
7752 return this->emitDummyPtr(D, E);
7755template <
class Emitter>
7763 if (!
C->destroyLocals())
7769template <
class Emitter>
7770unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7773 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7775 return Ty->getAsCXXRecordDecl();
7777 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7778 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7780 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7784template <
class Emitter>
7791 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7796 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7797 getFPOptions(E), E);
7799 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7800 getFPOptions(E), E);
7804 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7809 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7811 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7815 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7819 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7820 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7827template <
class Emitter>
7830 assert(FromT != ToT);
7833 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7835 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7837 return this->emitCast(FromT, ToT, E);
7841template <
class Emitter>
7842bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7846 return this->
discard(SubExpr);
7848 if (!this->visit(SubExpr))
7851 if (!this->emitConstUint8(0, SubExpr))
7853 return this->emitArrayElemPtrPopUint8(SubExpr);
7857 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7861template <
class Emitter>
7862bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7863 assert(!DiscardResult);
7867 if (!this->emitArrayElem(ElemT, 0, E))
7870 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7873 if (!this->emitCast(ElemT,
PT_Bool, E))
7878 LabelTy LabelTrue = this->getLabel();
7879 if (!this->jumpTrue(LabelTrue, E))
7882 if (!this->emitArrayElemPop(ElemT, 1, E))
7885 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7888 if (!this->emitCast(ElemT,
PT_Bool, E))
7892 LabelTy EndLabel = this->getLabel();
7893 this->jump(EndLabel, E);
7895 this->emitLabel(LabelTrue);
7896 if (!this->emitPopPtr(E))
7898 if (!this->emitConstBool(
true, E))
7901 this->fallthrough(EndLabel);
7902 this->emitLabel(EndLabel);
7907template <
class Emitter>
7908bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
7919 LHSIsComplex =
true;
7920 ElemT = classifyComplexElementType(LHS->
getType());
7921 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
7922 if (!this->visit(LHS))
7924 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
7927 LHSIsComplex =
false;
7929 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
7930 if (!this->visit(LHS))
7932 if (!this->emitSetLocal(LHST, LHSOffset, E))
7939 RHSIsComplex =
true;
7940 ElemT = classifyComplexElementType(RHS->
getType());
7941 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
7942 if (!this->visit(RHS))
7944 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
7947 RHSIsComplex =
false;
7949 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
7950 if (!this->visit(RHS))
7952 if (!this->emitSetLocal(RHST, RHSOffset, E))
7956 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
7957 bool IsComplex) ->
bool {
7959 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
7961 return this->emitArrayElemPop(ElemT, Index, E);
7963 return this->emitGetLocal(ElemT, LocalOffset, E);
7966 for (
unsigned I = 0; I != 2; ++I) {
7968 if (!getElem(LHSOffset, I, LHSIsComplex))
7970 if (!getElem(RHSOffset, I, RHSIsComplex))
7973 if (!this->emitEQ(ElemT, E))
7976 if (!this->emitCastBoolUint8(E))
7981 if (!this->emitAddUint8(E))
7983 if (!this->emitConstUint8(2, E))
7987 if (!this->emitEQUint8(E))
7990 if (!this->emitNEUint8(E))
7997 return this->emitCast(
PT_Bool, ResT, E);
8004template <
class Emitter>
8005bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
8008 assert(!
R->hasTrivialDtor());
8011 const Function *DtorFunc = getFunction(Dtor);
8014 assert(DtorFunc->hasThisPointer());
8015 assert(DtorFunc->getNumParams() == 1);
8016 return this->emitCall(DtorFunc, 0, Loc);
8021template <
class Emitter>
8022bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
8034 return this->emitPopPtr(Loc);
8036 for (ssize_t I = N - 1; I >= 1; --I) {
8037 if (!this->emitConstUint64(I, Loc))
8039 if (!this->emitArrayElemPtrUint64(Loc))
8041 if (!this->emitDestructionPop(ElemDesc, Loc))
8045 if (!this->emitConstUint64(0, Loc))
8047 if (!this->emitArrayElemPtrPopUint64(Loc))
8049 return this->emitDestructionPop(ElemDesc, Loc);
8054 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
8059template <
class Emitter>
8060bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E) {
8061 assert(!DiscardResult &&
"Should've been checked before");
8062 unsigned DummyID = P.getOrCreateDummy(D);
8064 if (!this->emitGetPtrGlobal(DummyID, E))
8072 return this->emitDecayPtr(
PT_Ptr, PT, E);
8078template <
class Emitter>
8079bool Compiler<Emitter>::emitFloat(
const APFloat &F,
const Expr *E) {
8081 return this->emitConstFloat(
Floating(F), E);
8083 APInt I = F.bitcastToAPInt();
8084 return this->emitConstFloat(
8085 Floating(
const_cast<uint64_t *
>(I.getRawData()),
8086 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
8097template <
class Emitter>
8098bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
8111 if (!this->emitGetPtrLocal(*LocalIndex, E))
8121 if (!this->visit(SubExpr))
8123 }
else if (
OptPrimType FromT = classify(SubExpr)) {
8124 unsigned TempOffset =
8125 allocateLocalPrimitive(SubExpr, *FromT,
true);
8126 if (!this->visit(SubExpr))
8128 if (!this->emitSetLocal(*FromT, TempOffset, E))
8130 if (!this->emitGetPtrLocal(TempOffset, E))
8137 if (!this->emitBitCast(E))
8139 return DiscardResult ? this->emitPopPtr(E) :
true;
8143 const llvm::fltSemantics *TargetSemantics =
nullptr;
8145 TargetSemantics = &Ctx.getFloatSemantics(ToType);
8151 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
8153 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
8154 ResultBitWidth, TargetSemantics,
8159 return this->emitPop(*ToT, E);
8168template <
class Emitter>
8169bool Compiler<Emitter>::emitHLSLAggregateSplat(
PrimType SrcT,
8174 unsigned NumElems = 0;
8177 NumElems = VT->getNumElements();
8178 ElemType = VT->getElementType();
8180 NumElems = MT->getNumElementsFlattened();
8181 ElemType = MT->getElementType();
8184 PrimType ElemT = classifyPrim(ElemType);
8185 for (
unsigned I = 0; I != NumElems; ++I) {
8186 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8188 if (!this->emitPrimCast(SrcT, ElemT, ElemType, E))
8190 if (!this->emitInitElem(ElemT, I, E))
8200 QualType ArrElemType = CAT->getElementType();
8201 unsigned ArrSize = CAT->getZExtSize();
8204 for (
unsigned I = 0; I != ArrSize; ++I) {
8205 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8207 if (!this->emitPrimCast(SrcT, *ElemT, ArrElemType, E))
8209 if (!this->emitInitElem(*ElemT, I, E))
8213 for (
unsigned I = 0; I != ArrSize; ++I) {
8214 if (!this->emitConstUint32(I, E))
8216 if (!this->emitArrayElemPtrUint32(E))
8218 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, ArrElemType, E))
8220 if (!this->emitFinishInitPop(E))
8230 const Record *
R = getRecord(DestType);
8234 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8236 const Record::Base *B =
R->getBase(BS.getType());
8238 if (!this->emitGetPtrBase(B->Offset, E))
8240 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, BS.getType(), E))
8242 if (!this->emitFinishInitPop(E))
8247 for (
const Record::Field &F :
R->fields()) {
8248 if (F.isUnnamedBitField())
8251 QualType FieldType = F.Decl->getType();
8253 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8255 if (!this->emitPrimCast(SrcT, *FieldT, FieldType, E))
8257 if (F.isBitField()) {
8258 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8261 if (!this->emitInitField(*FieldT, F.Offset, E))
8265 if (!this->emitGetPtrField(F.Offset, E))
8267 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, FieldType, E))
8269 if (!this->emitPopPtr(E))
8282template <
class Emitter>
8283unsigned Compiler<Emitter>::countHLSLFlatElements(
QualType Ty) {
8286 return VT->getNumElements();
8288 return MT->getNumElementsFlattened();
8292 return CAT->getZExtSize() * countHLSLFlatElements(CAT->getElementType());
8296 const Record *
R = getRecord(Ty);
8300 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8302 Count += countHLSLFlatElements(BS.getType());
8304 for (
const Record::Field &F :
R->fields()) {
8305 if (F.isUnnamedBitField())
8307 Count += countHLSLFlatElements(F.Decl->getType());
8312 if (canClassify(Ty))
8321template <
class Emitter>
8322bool Compiler<Emitter>::emitHLSLFlattenAggregate(
8323 QualType SrcType,
unsigned SrcOffset,
8328 auto saveToLocal = [&](
PrimType T) ->
bool {
8329 unsigned Offset = allocateLocalPrimitive(E, T,
true);
8330 if (!this->emitSetLocal(T, Offset, E))
8332 Elements.push_back({Offset, T});
8338 unsigned Offset = allocateLocalPrimitive(E,
PT_Ptr,
true);
8339 if (!this->emitSetLocal(
PT_Ptr, Offset, E))
8340 return std::nullopt;
8345 unsigned NumElems = 0;
8348 NumElems = VT->getNumElements();
8349 ElemType = VT->getElementType();
8351 NumElems = MT->getNumElementsFlattened();
8352 ElemType = MT->getElementType();
8355 PrimType ElemT = classifyPrim(ElemType);
8356 for (
unsigned I = 0; I != NumElems && Elements.size() < MaxElements; ++I) {
8357 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8359 if (!this->emitArrayElemPop(ElemT, I, E))
8361 if (!saveToLocal(ElemT))
8371 QualType ArrElemType = CAT->getElementType();
8372 unsigned ArrSize = CAT->getZExtSize();
8375 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8376 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8378 if (!this->emitArrayElemPop(*ElemT, I, E))
8380 if (!saveToLocal(*ElemT))
8384 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8385 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8387 if (!this->emitConstUint32(I, E))
8389 if (!this->emitArrayElemPtrPopUint32(E))
8394 if (!emitHLSLFlattenAggregate(ArrElemType, *ElemPtrOffset, Elements,
8405 const Record *
R = getRecord(SrcType);
8409 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8411 if (Elements.size() >= MaxElements)
8413 const Record::Base *B =
R->getBase(BS.getType());
8415 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8417 if (!this->emitGetPtrBasePop(B->Offset,
false, E))
8422 if (!emitHLSLFlattenAggregate(BS.getType(), *BasePtrOffset, Elements,
8428 for (
const Record::Field &F :
R->fields()) {
8429 if (Elements.size() >= MaxElements)
8431 if (F.isUnnamedBitField())
8434 QualType FieldType = F.Decl->getType();
8435 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8437 if (!this->emitGetPtrFieldPop(F.Offset, E))
8441 if (!this->emitLoadPop(*FieldT, E))
8443 if (!saveToLocal(*FieldT))
8447 if (!FieldPtrOffset)
8449 if (!emitHLSLFlattenAggregate(FieldType, *FieldPtrOffset, Elements,
8465template <
class Emitter>
8466bool Compiler<Emitter>::emitHLSLConstructAggregate(
8472 const auto &Src = Elements[ElemIdx++];
8473 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
8475 return this->emitPrimCast(Src.Type, DestT, DestQT, E);
8479 unsigned NumElems = 0;
8482 NumElems = VT->getNumElements();
8483 ElemType = VT->getElementType();
8485 NumElems = MT->getNumElementsFlattened();
8486 ElemType = MT->getElementType();
8489 PrimType DestElemT = classifyPrim(ElemType);
8490 for (
unsigned I = 0; I != NumElems; ++I) {
8491 if (!loadAndCast(DestElemT, ElemType))
8493 if (!this->emitInitElem(DestElemT, I, E))
8503 QualType ArrElemType = CAT->getElementType();
8504 unsigned ArrSize = CAT->getZExtSize();
8507 for (
unsigned I = 0; I != ArrSize; ++I) {
8508 if (!loadAndCast(*ElemT, ArrElemType))
8510 if (!this->emitInitElem(*ElemT, I, E))
8514 for (
unsigned I = 0; I != ArrSize; ++I) {
8515 if (!this->emitConstUint32(I, E))
8517 if (!this->emitArrayElemPtrUint32(E))
8519 if (!emitHLSLConstructAggregate(ArrElemType, Elements, ElemIdx, E))
8521 if (!this->emitFinishInitPop(E))
8531 const Record *
R = getRecord(DestType);
8535 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8537 const Record::Base *B =
R->getBase(BS.getType());
8539 if (!this->emitGetPtrBase(B->Offset, E))
8541 if (!emitHLSLConstructAggregate(BS.getType(), Elements, ElemIdx, E))
8543 if (!this->emitFinishInitPop(E))
8548 for (
const Record::Field &F :
R->fields()) {
8549 if (F.isUnnamedBitField())
8552 QualType FieldType = F.Decl->getType();
8554 if (!loadAndCast(*FieldT, FieldType))
8556 if (F.isBitField()) {
8557 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8560 if (!this->emitInitField(*FieldT, F.Offset, E))
8564 if (!this->emitGetPtrField(F.Offset, E))
8566 if (!emitHLSLConstructAggregate(FieldType, Elements, ElemIdx, E))
8568 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() const
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.
ValueDecl * getExtendingDecl()
Get the declaration which triggered the lifetime-extension of this temporary, if any.
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 visitLValueExpr(const Expr *E, bool DestroyToplevelScope) override
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()