29 if (
const auto *CE = dyn_cast_if_present<ConstantExpr>(E);
30 CE && CE->hasAPValueResult() &&
32 return CE->getResultAsAPSInt().getBoolValue();
43 OldInitializingDecl(
Ctx->InitializingDecl) {
44 Ctx->InitializingDecl = VD;
49 this->
Ctx->InitializingDecl = OldInitializingDecl;
50 this->
Ctx->InitStack.pop_back();
63 bool NewInitializing,
bool NewToLValue)
64 : Ctx(Ctx), OldDiscardResult(Ctx->DiscardResult),
65 OldInitializing(Ctx->
Initializing), OldToLValue(Ctx->ToLValue) {
66 Ctx->DiscardResult = NewDiscardResult;
67 Ctx->Initializing = NewInitializing;
68 Ctx->ToLValue = NewToLValue;
72 Ctx->DiscardResult = OldDiscardResult;
73 Ctx->Initializing = OldInitializing;
74 Ctx->ToLValue = OldToLValue;
81 bool OldDiscardResult;
86template <
class Emitter>
90 return Ctx->emitThis(E);
93 return Ctx->emitGetPtrFieldPop(
Offset, E);
95 return Ctx->emitGetPtrLocal(
Offset, E);
99 if (!Ctx->emitConstUint32(
Offset, E))
101 return Ctx->emitArrayElemPtrPopUint32(E);
103 return Ctx->emitRVOPtr(E);
107 llvm_unreachable(
"Unhandled InitLink kind");
123 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
124 assert(LI.Name != Name);
127 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel, ContinueLabel,
148 : Ctx(Ctx), OldCaseLabels(
std::move(this->Ctx->CaseLabels)) {
150 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
151 assert(LI.Name != Name);
154 this->Ctx->CaseLabels = std::move(CaseLabels);
155 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel,
157 DefaultLabel, Ctx->VarScope);
161 this->Ctx->CaseLabels = std::move(OldCaseLabels);
162 this->Ctx->LabelInfoStack.pop_back();
167 CaseMap OldCaseLabels;
173 Ctx->InStmtExpr =
true;
191 : Ctx(Ctx), OldFlag(Ctx->LocOverride), Enabled(Enabled) {
194 Ctx->LocOverride = NewValue;
199 Ctx->LocOverride = OldFlag;
204 std::optional<SourceInfo> OldFlag;
211template <
class Emitter>
219 case CK_LValueToRValue: {
230 if (
const auto *DRE = dyn_cast<DeclRefExpr>(SubExpr)) {
236 if (
auto GlobalIndex =
P.getGlobal(D))
237 return this->emitGetGlobal(*SubExprT, *GlobalIndex, CE);
238 }
else if (
auto It =
Locals.find(D); It !=
Locals.end()) {
239 return this->emitGetLocal(*SubExprT, It->second.Offset, CE);
240 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
241 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
242 return this->emitGetParam(*SubExprT, It->second.Offset, CE);
254 if (!this->emitGetPtrLocal(*LocalIndex, CE))
258 if (!this->
visit(SubExpr))
262 return this->emitLoadPop(*SubExprT, CE);
267 return this->emitMemcpy(CE);
270 case CK_DerivedToBaseMemberPointer: {
276 unsigned DerivedOffset =
278 FromMP->getMostRecentCXXRecordDecl());
283 return this->emitGetMemberPtrBasePop(DerivedOffset, CE);
286 case CK_BaseToDerivedMemberPointer: {
292 unsigned DerivedOffset =
293 Ctx.collectBaseOffset(FromMP->getMostRecentCXXRecordDecl(),
298 return this->emitGetMemberPtrBasePop(-DerivedOffset, CE);
301 case CK_UncheckedDerivedToBase:
302 case CK_DerivedToBase: {
307 if (
const auto *PT = dyn_cast<PointerType>(Ty))
308 return PT->getPointeeType()->getAsCXXRecordDecl();
309 return Ty->getAsCXXRecordDecl();
316 if (B->isVirtual()) {
317 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), CE))
319 CurType = B->getType();
321 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
322 if (!this->emitGetPtrBasePop(
325 CurType = B->getType();
332 case CK_BaseToDerived: {
335 unsigned DerivedOffset =
341 return this->emitGetPtrDerivedPop(DerivedOffset,
346 case CK_FloatingCast: {
351 if (!this->
visit(SubExpr))
353 const auto *TargetSemantics = &
Ctx.getFloatSemantics(CE->
getType());
357 case CK_IntegralToFloating: {
360 if (!this->
visit(SubExpr))
362 const auto *TargetSemantics = &
Ctx.getFloatSemantics(CE->
getType());
363 return this->emitCastIntegralFloating(
364 classifyPrim(SubExpr), TargetSemantics, getFPOptions(CE), CE);
367 case CK_FloatingToBoolean: {
371 if (
const auto *FL = dyn_cast<FloatingLiteral>(SubExpr))
372 return this->emitConstBool(FL->getValue().isNonZero(), CE);
373 if (!this->
visit(SubExpr))
375 return this->emitCastFloatingIntegralBool(getFPOptions(CE), CE);
378 case CK_FloatingToIntegral: {
381 if (!this->
visit(SubExpr))
385 return this->emitCastFloatingIntegralAP(
Ctx.getBitWidth(CE->
getType()),
386 getFPOptions(CE), CE);
388 return this->emitCastFloatingIntegralAPS(
Ctx.getBitWidth(CE->
getType()),
389 getFPOptions(CE), CE);
391 return this->emitCastFloatingIntegral(ToT, getFPOptions(CE), CE);
394 case CK_NullToPointer:
395 case CK_NullToMemberPointer: {
400 if (!PointeeType.
isNull()) {
402 Desc =
P.createDescriptor(SubExpr, *
T);
404 Desc =
P.createDescriptor(SubExpr, PointeeType.
getTypePtr(),
408 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(CE->
getType());
412 case CK_PointerToIntegral: {
413 if (!this->
visit(SubExpr))
419 if (!this->emitDecayPtr(FromT,
PT_Ptr, CE))
425 return this->emitCastPointerIntegralAP(
Ctx.getBitWidth(CE->
getType()),
428 return this->emitCastPointerIntegralAPS(
Ctx.getBitWidth(CE->
getType()),
430 return this->emitCastPointerIntegral(
T, CE);
433 case CK_ArrayToPointerDecay: {
434 if (!this->
visit(SubExpr))
436 return this->emitArrayDecay(CE);
439 case CK_IntegralToPointer: {
441 assert(IntType->isIntegralOrEnumerationType());
442 if (!this->
visit(SubExpr))
450 Desc =
P.createDescriptor(SubExpr, *
T);
457 if (!this->emitGetIntPtr(
T, Desc, CE))
465 return this->emitDecayPtr(
PT_Ptr, DestPtrT, CE);
468 case CK_AtomicToNonAtomic:
469 case CK_ConstructorConversion:
470 case CK_FunctionToPointerDecay:
471 case CK_NonAtomicToAtomic:
473 case CK_UserDefinedConversion:
474 case CK_AddressSpaceConversion:
475 case CK_CPointerToObjCPointerCast:
490 return this->emitBuiltinBitCast(CE);
505 if (!this->
visit(SubExpr))
513 return this->emitFnPtrCast(CE);
520 if (!this->
visit(SubExpr))
522 return this->emitDecayPtr(*FromT, *ToT, CE);
524 case CK_IntegralToBoolean:
525 case CK_FixedPointToBoolean: {
531 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr))
532 return this->emitConst(IL->getValue(), CE);
533 if (!this->
visit(SubExpr))
538 case CK_BooleanToSignedIntegral:
539 case CK_IntegralCast: {
546 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr)) {
551 if (!this->emitConst(IL->getValue(), SubExpr))
554 if (!this->
visit(SubExpr))
562 if (!ED->isFixed()) {
563 if (!this->emitCheckEnumValue(*FromT, ED, CE))
569 if (!this->emitCastAP(*FromT,
Ctx.getBitWidth(CE->
getType()), CE))
572 if (!this->emitCastAPS(*FromT,
Ctx.getBitWidth(CE->
getType()), CE))
577 if (!this->emitCast(*FromT, *ToT, CE))
580 if (CE->
getCastKind() == CK_BooleanToSignedIntegral)
581 return this->emitNeg(*ToT, CE);
585 case CK_PointerToBoolean:
586 case CK_MemberPointerToBoolean: {
589 if (!this->
visit(SubExpr))
591 return this->emitIsNonNull(PtrT, CE);
594 case CK_IntegralComplexToBoolean:
595 case CK_FloatingComplexToBoolean: {
596 if (!this->
visit(SubExpr))
598 return this->emitComplexBoolCast(SubExpr);
601 case CK_IntegralComplexToReal:
602 case CK_FloatingComplexToReal:
603 return this->emitComplexReal(SubExpr);
605 case CK_IntegralRealToComplex:
606 case CK_FloatingRealToComplex: {
613 if (!this->emitGetPtrLocal(*LocalIndex, CE))
622 if (!this->visitZeroInitializer(
T, SubExpr->
getType(), SubExpr))
624 return this->emitInitElem(
T, 1, SubExpr);
627 case CK_IntegralComplexCast:
628 case CK_FloatingComplexCast:
629 case CK_IntegralComplexToFloatingComplex:
630 case CK_FloatingComplexToIntegralComplex: {
637 if (!this->emitGetPtrLocal(*LocalIndex, CE))
644 unsigned SubExprOffset =
646 if (!this->
visit(SubExpr))
648 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, CE))
656 for (
unsigned I = 0; I != 2; ++I) {
657 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, CE))
659 if (!this->emitArrayElemPop(SourceElemT, I, CE))
663 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, CE))
667 if (!this->emitInitElem(DestElemT, I, CE))
673 case CK_VectorSplat: {
682 if (!this->emitGetPtrLocal(*LocalIndex, CE))
688 unsigned ElemOffset =
692 if (!this->
visit(SubExpr))
697 if (!this->emitSetLocal(ElemT, ElemOffset, CE))
700 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
701 if (!this->emitGetLocal(ElemT, ElemOffset, CE))
703 if (!this->emitInitElem(ElemT, I, CE))
710 case CK_HLSLVectorTruncation: {
715 if (!this->
visit(SubExpr))
717 return this->emitArrayElemPop(*ResultT, 0, CE);
726 if (!this->emitGetPtrLocal(*LocalIndex, CE))
731 if (!this->
visit(SubExpr))
733 return this->emitCopyArray(classifyVectorElementType(CE->
getType()), 0, 0,
737 case CK_IntegralToFixedPoint: {
738 if (!this->
visit(SubExpr))
742 Ctx.getASTContext().getFixedPointSemantics(CE->
getType()).toOpaqueInt();
746 case CK_FloatingToFixedPoint: {
747 if (!this->
visit(SubExpr))
751 Ctx.getASTContext().getFixedPointSemantics(CE->
getType()).toOpaqueInt();
752 return this->emitCastFloatingFixedPoint(Sem, CE);
754 case CK_FixedPointToFloating: {
755 if (!this->
visit(SubExpr))
757 const auto *TargetSemantics = &
Ctx.getFloatSemantics(CE->
getType());
758 return this->emitCastFixedPointFloating(TargetSemantics, CE);
760 case CK_FixedPointToIntegral: {
761 if (!this->
visit(SubExpr))
765 case CK_FixedPointCast: {
766 if (!this->
visit(SubExpr))
769 Ctx.getASTContext().getFixedPointSemantics(CE->
getType()).toOpaqueInt();
770 return this->emitCastFixedPoint(Sem, CE);
782 return this->emitInvalid(CE);
784 llvm_unreachable(
"Unhandled clang::CastKind enum");
787template <
class Emitter>
789 return this->emitBuiltinBitCast(E);
792template <
class Emitter>
797 return this->emitConst(
LE->getValue(),
LE);
800template <
class Emitter>
806 return this->emitFloat(F, E);
809template <
class Emitter>
819 if (!this->emitGetPtrLocal(*LocalIndex, E))
826 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
828 if (!this->emitInitElem(SubExprT, 0, SubExpr))
833template <
class Emitter>
841 auto Sem =
Ctx.getASTContext().getFixedPointSemantics(E->
getType());
846template <
class Emitter>
851template <
class Emitter>
878 return this->emitComplexComparison(LHS, RHS, BO);
883 if (!this->
visit(LHS))
886 if (!this->
visit(RHS))
889 if (!this->emitToMemberPtr(BO))
895 if (!this->emitCastMemberPtrPtr(BO))
912 Ctx.getASTContext().CompCategories.lookupInfoForType(BO->
getType());
918 if (!this->emitGetPtrLocal(*ResultIndex, BO))
925 return this->emitCMP3(*
LT, CmpInfo, BO);
928 if (!
LT || !RT || !
T)
938 return this->visitAssignment(LHS, RHS, BO);
945 auto MaybeCastToBool = [
this,
T, BO](
bool Result) {
949 return this->emitPopBool(BO);
951 return this->emitCast(
PT_Bool, *
T, BO);
955 auto Discard = [
this,
T, BO](
bool Result) {
963 return MaybeCastToBool(this->emitEQ(*
LT, BO));
965 return MaybeCastToBool(this->emitNE(*
LT, BO));
967 return MaybeCastToBool(this->emitLT(*
LT, BO));
969 return MaybeCastToBool(this->emitLE(*
LT, BO));
971 return MaybeCastToBool(this->emitGT(*
LT, BO));
973 return MaybeCastToBool(this->emitGE(*
LT, BO));
976 return Discard(this->emitSubf(getFPOptions(BO), BO));
977 return Discard(this->emitSub(*
T, BO));
980 return Discard(this->emitAddf(getFPOptions(BO), BO));
981 return Discard(this->emitAdd(*
T, BO));
984 return Discard(this->emitMulf(getFPOptions(BO), BO));
985 return Discard(this->emitMul(*
T, BO));
987 return Discard(this->emitRem(*
T, BO));
990 return Discard(this->emitDivf(getFPOptions(BO), BO));
991 return Discard(this->emitDiv(*
T, BO));
993 return Discard(this->emitBitAnd(*
T, BO));
995 return Discard(this->emitBitOr(*
T, BO));
997 return Discard(this->emitShl(*
LT, *RT, BO));
999 return Discard(this->emitShr(*
LT, *RT, BO));
1001 return Discard(this->emitBitXor(*
T, BO));
1004 llvm_unreachable(
"Already handled earlier");
1009 llvm_unreachable(
"Unhandled binary op");
1014template <
class Emitter>
1020 if ((Op != BO_Add && Op != BO_Sub) ||
1031 auto visitAsPointer = [&](
const Expr *E,
PrimType T) ->
bool {
1032 if (!this->
visit(E))
1035 return this->emitDecayPtr(
T,
PT_Ptr, E);
1044 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1052 ElemTypeSize =
Ctx.getASTContext().getTypeSizeInChars(ElemType);
1055 if (!this->emitSubPtr(IntT, ElemTypeSize.
isZero(), E))
1062 if (!visitAsPointer(RHS, *RT))
1064 if (!this->
visit(LHS))
1068 if (!visitAsPointer(LHS, *
LT))
1070 if (!this->
visit(RHS))
1080 if (!this->emitAddOffset(OffsetType, E))
1088 if (!this->emitSubOffset(OffsetType, E))
1099template <
class Emitter>
1109 LabelTy LabelTrue = this->getLabel();
1110 LabelTy LabelEnd = this->getLabel();
1114 if (!this->jumpTrue(LabelTrue))
1119 if (!this->jump(LabelEnd))
1122 this->emitLabel(LabelTrue);
1123 this->emitConstBool(
true, E);
1124 this->fallthrough(LabelEnd);
1125 this->emitLabel(LabelEnd);
1128 assert(Op == BO_LAnd);
1131 LabelTy LabelFalse = this->getLabel();
1132 LabelTy LabelEnd = this->getLabel();
1136 if (!this->jumpFalse(LabelFalse))
1141 if (!this->jump(LabelEnd))
1144 this->emitLabel(LabelFalse);
1145 this->emitConstBool(
false, E);
1146 this->fallthrough(LabelEnd);
1147 this->emitLabel(LabelEnd);
1151 return this->emitPopBool(E);
1156 return this->emitCast(
PT_Bool, *
T, E);
1160template <
class Emitter>
1167 if (!this->emitGetPtrLocal(*LocalIndex, E))
1176 PrimType ResultElemT = this->classifyComplexElementType(E->
getType());
1177 unsigned ResultOffset = ~0u;
1183 if (!this->emitDupPtr(E))
1185 if (!this->emitSetLocal(
PT_Ptr, ResultOffset, E))
1190 LHSType = AT->getValueType();
1193 RHSType = AT->getValueType();
1202 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1207 if (!this->
visit(LHS))
1209 if (!this->
visit(RHS))
1211 return this->emitMulc(ElemT, E);
1214 if (Op == BO_Div && RHSIsComplex) {
1221 if (!LHSIsComplex) {
1226 LHSOffset = *LocalIndex;
1228 if (!this->emitGetPtrLocal(LHSOffset, E))
1231 if (!this->
visit(LHS))
1234 if (!this->emitInitElem(ElemT, 0, E))
1237 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1239 if (!this->emitInitElem(ElemT, 1, E))
1242 if (!this->
visit(LHS))
1246 if (!this->
visit(RHS))
1248 return this->emitDivc(ElemT, E);
1254 if (!this->
visit(LHS))
1256 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1261 if (!this->
visit(LHS))
1263 if (!this->emitSetLocal(LHST, LHSOffset, E))
1271 if (!this->
visit(RHS))
1273 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1278 if (!this->
visit(RHS))
1280 if (!this->emitSetLocal(RHST, RHSOffset, E))
1287 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1288 unsigned ElemIndex,
unsigned Offset,
1289 const Expr *E) ->
bool {
1291 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1293 return this->emitArrayElemPop(classifyComplexElementType(E->
getType()),
1296 if (ElemIndex == 0 || !LoadZero)
1303 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1306 if (!this->emitGetLocal(
PT_Ptr, ResultOffset, E))
1313 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1316 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1319 if (!this->emitAddf(getFPOptions(E), E))
1322 if (!this->emitAdd(ResultElemT, E))
1327 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1330 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1333 if (!this->emitSubf(getFPOptions(E), E))
1336 if (!this->emitSub(ResultElemT, E))
1341 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1344 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1348 if (!this->emitMulf(getFPOptions(E), E))
1351 if (!this->emitMul(ResultElemT, E))
1356 assert(!RHSIsComplex);
1357 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1360 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1364 if (!this->emitDivf(getFPOptions(E), E))
1367 if (!this->emitDiv(ResultElemT, E))
1378 if (!this->emitInitElemPop(ResultElemT, ElemIndex, E))
1381 if (!this->emitPop(ResultElemT, E))
1388template <
class Emitter>
1393 "Comma op should be handled in VisitBinaryOperator");
1407 if (!this->emitGetPtrLocal(*LocalIndex, E))
1421 assert(
Ctx.getASTContext().hasSameUnqualifiedType(
1424 if (!this->
visit(LHS))
1426 if (!this->
visit(RHS))
1428 if (!this->emitCopyArray(ElemT, 0, 0, VecTy->getNumElements(), E))
1431 return this->emitPopPtr(E);
1436 unsigned LHSOffset =
1438 if (!this->
visit(LHS))
1440 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1444 unsigned RHSOffset =
1446 if (!this->
visit(RHS))
1448 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1460 if (NeedIntPromot) {
1462 Ctx.getASTContext().getPromotedIntegerType(
Ctx.getASTContext().BoolTy);
1467 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1468 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1470 if (!this->emitArrayElemPop(ElemT, Index, E))
1473 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
1475 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1477 }
else if (NeedIntPromot) {
1478 if (!this->emitPrimCast(ElemT, PromotT, PromotTy, E))
1484#define EMIT_ARITH_OP(OP) \
1486 if (ElemT == PT_Float) { \
1487 if (!this->emit##OP##f(getFPOptions(E), E)) \
1490 if (!this->emit##OP(ElemT, E)) \
1496 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1497 if (!getElem(LHSOffset, ElemT, I))
1499 if (!getElem(RHSOffset, RHSElemT, I))
1511 if (!this->emitRem(ElemT, E))
1515 if (!this->emitBitAnd(OpT, E))
1519 if (!this->emitBitOr(OpT, E))
1523 if (!this->emitBitXor(OpT, E))
1527 if (!this->emitShl(OpT, RHSElemT, E))
1531 if (!this->emitShr(OpT, RHSElemT, E))
1535 if (!this->emitEQ(ElemT, E))
1539 if (!this->emitNE(ElemT, E))
1543 if (!this->emitLE(ElemT, E))
1547 if (!this->emitLT(ElemT, E))
1551 if (!this->emitGE(ElemT, E))
1555 if (!this->emitGT(ElemT, E))
1560 if (!this->emitBitAnd(ResultElemT, E))
1565 if (!this->emitBitOr(ResultElemT, E))
1569 return this->emitInvalid(E);
1578 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1580 if (!this->emitNeg(ResultElemT, E))
1586 if (NeedIntPromot &&
1587 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(), E))
1591 if (!this->emitInitElem(ResultElemT, I, E))
1600template <
class Emitter>
1610 auto LHSSemaInt = LHSSema.toOpaqueInt();
1612 auto RHSSemaInt = RHSSema.toOpaqueInt();
1614 if (!this->
visit(LHS))
1622 if (!this->
visit(RHS))
1631 auto ConvertResult = [&](
bool R) ->
bool {
1635 auto CommonSema = LHSSema.getCommonSemantics(RHSSema).toOpaqueInt();
1636 if (ResultSema != CommonSema)
1637 return this->emitCastFixedPoint(ResultSema, E);
1641 auto MaybeCastToBool = [&](
bool Result) {
1646 return this->emitPop(
T, E);
1648 return this->emitCast(
PT_Bool,
T, E);
1654 return MaybeCastToBool(this->emitEQFixedPoint(E));
1656 return MaybeCastToBool(this->emitNEFixedPoint(E));
1658 return MaybeCastToBool(this->emitLTFixedPoint(E));
1660 return MaybeCastToBool(this->emitLEFixedPoint(E));
1662 return MaybeCastToBool(this->emitGTFixedPoint(E));
1664 return MaybeCastToBool(this->emitGEFixedPoint(E));
1666 return ConvertResult(this->emitAddFixedPoint(E));
1668 return ConvertResult(this->emitSubFixedPoint(E));
1670 return ConvertResult(this->emitMulFixedPoint(E));
1672 return ConvertResult(this->emitDivFixedPoint(E));
1674 return ConvertResult(this->emitShiftFixedPoint(
true, E));
1676 return ConvertResult(this->emitShiftFixedPoint(
false, E));
1679 return this->emitInvalid(E);
1682 llvm_unreachable(
"unhandled binop opcode");
1685template <
class Emitter>
1694 if (!this->
visit(SubExpr))
1696 return this->emitNegFixedPoint(E);
1701 llvm_unreachable(
"Unhandled unary opcode");
1704template <
class Emitter>
1710 return this->visitZeroInitializer(*
T, QT, E);
1718 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1719 CXXRD && CXXRD->getNumVBases() > 0) {
1729 return this->visitZeroRecordInitializer(R, E);
1736 return this->visitZeroArrayInitializer(QT, E);
1740 QualType ElemQT = ComplexTy->getElementType();
1742 for (
unsigned I = 0; I < 2; ++I) {
1743 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1745 if (!this->emitInitElem(ElemT, I, E))
1752 unsigned NumVecElements = VecT->getNumElements();
1753 QualType ElemQT = VecT->getElementType();
1756 for (
unsigned I = 0; I < NumVecElements; ++I) {
1757 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1759 if (!this->emitInitElem(ElemT, I, E))
1768template <
class Emitter>
1778 for (
const Expr *SubExpr : {LHS, RHS}) {
1779 if (!this->
visit(SubExpr)) {
1786 if (SubExpr ==
Base &&
Base->getType()->isPointerType()) {
1787 if (!this->emitExpandPtr(E))
1798 return this->emitError(E);
1801 if (!this->emitFlip(
PT_Ptr, *IndexT, E))
1805 if (!this->emitArrayElemPtrPop(*IndexT, E))
1808 return this->emitPopPtr(E);
1814 return this->emitLoadPop(*
T, E);
1817template <
class Emitter>
1819 const Expr *ArrayFiller,
const Expr *E) {
1824 QT = AT->getValueType();
1827 if (Inits.size() == 0)
1829 return this->emitInvalid(E);
1844 if (Inits.size() == 0)
1845 return this->visitZeroInitializer(*
T, QT, E);
1846 assert(Inits.size() == 1);
1853 if (Inits.size() == 1 && E->
getType() == Inits[0]->getType())
1859 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
1866 bool BitField = FieldToInit->isBitField();
1868 return this->emitInitBitFieldActivate(
T, FieldToInit, E);
1870 return this->emitInitBitField(
T, FieldToInit, E);
1872 return this->emitInitFieldActivate(
T, FieldToInit->Offset, E);
1873 return this->emitInitField(
T, FieldToInit->Offset, E);
1876 auto initCompositeField = [=](
const Record::Field *FieldToInit,
1884 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
1887 if (
Activate && !this->emitActivate(E))
1892 return this->emitPopPtr(E);
1896 if (Inits.size() == 0) {
1897 if (!this->visitZeroRecordInitializer(R, E))
1902 if (
const auto *ILE = dyn_cast<InitListExpr>(E))
1903 FToInit = ILE->getInitializedFieldInUnion();
1907 const Record::Field *FieldToInit = R->
getField(FToInit);
1909 if (!initPrimitiveField(FieldToInit,
Init, *
T,
true))
1912 if (!initCompositeField(FieldToInit,
Init,
true))
1916 return this->emitFinishInit(E);
1920 unsigned InitIndex = 0;
1923 while (InitIndex < R->getNumFields() &&
1928 const Record::Field *FieldToInit = R->
getField(InitIndex);
1929 if (!initPrimitiveField(FieldToInit,
Init, *
T))
1934 if (
const Record::Base *B = R->
getBase(
Init->getType())) {
1935 if (!this->emitGetPtrBase(B->Offset,
Init))
1941 if (!this->emitFinishInitPop(E))
1946 const Record::Field *FieldToInit = R->
getField(InitIndex);
1947 if (!initCompositeField(FieldToInit,
Init))
1953 return this->emitFinishInit(E);
1957 if (Inits.size() == 1 && QT == Inits[0]->getType())
1961 Ctx.getASTContext().getAsConstantArrayType(QT);
1964 if (!this->emitCheckArraySize(NumElems, E))
1968 unsigned ElementIndex = 0;
1970 if (
const auto *EmbedS =
1971 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
1979 if (!this->emitCastIntegralFloating(
classifyPrim(IL), Sem,
1980 getFPOptions(E), E))
1986 return this->emitInitElem(TargetT, ElemIndex, IL);
1988 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
2000 for (; ElementIndex != NumElems; ++ElementIndex) {
2006 return this->emitFinishInit(E);
2010 unsigned NumInits = Inits.size();
2015 QualType ElemQT = ComplexTy->getElementType();
2017 if (NumInits == 0) {
2019 for (
unsigned I = 0; I < 2; ++I) {
2020 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2022 if (!this->emitInitElem(ElemT, I, E))
2025 }
else if (NumInits == 2) {
2026 unsigned InitIndex = 0;
2031 if (!this->emitInitElem(ElemT, InitIndex, E))
2040 unsigned NumVecElements = VecT->getNumElements();
2041 assert(NumVecElements >= Inits.size());
2043 QualType ElemQT = VecT->getElementType();
2047 unsigned InitIndex = 0;
2054 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
2055 if (!this->emitCopyArray(ElemT, 0, InitIndex,
2056 InitVecT->getNumElements(), E))
2058 InitIndex += InitVecT->getNumElements();
2060 if (!this->emitInitElem(ElemT, InitIndex, E))
2066 assert(InitIndex <= NumVecElements);
2069 for (; InitIndex != NumVecElements; ++InitIndex) {
2070 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2072 if (!this->emitInitElem(ElemT, InitIndex, E))
2083template <
class Emitter>
2090 return this->emitInitElem(*InitT, ElemIndex,
Init);
2096 if (!this->emitConstUint32(ElemIndex,
Init))
2098 if (!this->emitArrayElemPtrUint32(
Init))
2102 return this->emitFinishInitPop(
Init);
2105template <
class Emitter>
2108 bool Activate,
bool IsOperatorCall) {
2110 llvm::BitVector NonNullArgs;
2111 if (FuncDecl && FuncDecl->
hasAttr<NonNullAttr>())
2114 bool ExplicitMemberFn =
false;
2115 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl))
2116 ExplicitMemberFn = MD->isExplicitObjectMemberFunction();
2118 unsigned ArgIndex = 0;
2119 for (
const Expr *Arg : Args) {
2121 if (!this->
visit(Arg))
2129 unsigned DeclIndex = ArgIndex - IsOperatorCall + ExplicitMemberFn;
2130 if (DeclIndex < FuncDecl->getNumParams())
2131 Source = FuncDecl->
getParamDecl(ArgIndex - IsOperatorCall +
2141 if (!this->emitGetPtrLocal(*LocalIndex, Arg))
2149 if (!this->emitActivate(Arg))
2153 if (!NonNullArgs.empty() && NonNullArgs[ArgIndex]) {
2156 if (!this->emitCheckNonNullArg(ArgT, Arg))
2167template <
class Emitter>
2172template <
class Emitter>
2178template <
class Emitter>
2184template <
class Emitter>
2200template <
class Emitter>
2202 auto It = E->
begin();
2203 return this->
visit(*It);
2208 bool AlignOfReturnsPreferred =
2209 ASTCtx.
getLangOpts().getClangABICompat() <= LangOptions::ClangABI::Ver7;
2215 T = Ref->getPointeeType();
2217 if (
T.getQualifiers().hasUnaligned())
2223 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2229template <
class Emitter>
2235 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2241 ArgType = Ref->getPointeeType();
2247 if (
ArgType->isDependentType() || !
ArgType->isConstantSizeType())
2248 return this->emitInvalid(E);
2250 if (Kind == UETT_SizeOf)
2259 return this->emitConst(Size.getQuantity(), E);
2262 if (Kind == UETT_CountOf) {
2268 if (
const auto *CAT =
2272 return this->emitConst(CAT->getSize(), E);
2282 if (VAT->getElementType()->isArrayType()) {
2283 std::optional<APSInt> Res =
2285 ? VAT->getSizeExpr()->getIntegerConstantExpr(ASTCtx)
2290 return this->emitConst(*Res, E);
2295 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2312 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2315 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2325 return this->emitConst(Size.getQuantity(), E);
2328 if (Kind == UETT_VectorElements) {
2330 return this->emitConst(VT->getNumElements(), E);
2332 return this->emitSizelessVectorElementSize(E);
2335 if (Kind == UETT_VecStep) {
2337 unsigned N = VT->getNumElements();
2344 return this->emitConst(N, E);
2346 return this->emitConst(1, E);
2349 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2356 if (Kind == UETT_PtrAuthTypeDiscriminator) {
2358 return this->emitInvalid(E);
2360 return this->emitConst(
2361 const_cast<ASTContext &
>(ASTCtx).getPointerAuthTypeDiscriminator(
2369template <
class Emitter>
2380 const auto maybeLoadValue = [&]() ->
bool {
2384 return this->emitLoadPop(*
T, E);
2388 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2392 if (
auto GlobalIndex =
P.getGlobal(VD))
2393 return this->emitGetPtrGlobal(*GlobalIndex, E) && maybeLoadValue();
2398 if (!this->
discard(Base) && !this->emitSideEffect(E))
2413 const Record::Field *F = R->
getField(FD);
2415 if (F->Decl->getType()->isReferenceType())
2416 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2417 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2420template <
class Emitter>
2429template <
class Emitter>
2448 for (
size_t I = 0; I != Size; ++I) {
2460template <
class Emitter>
2471 return this->emitGetLocal(SubExprT, It->second, E);
2473 if (!this->
visit(SourceExpr))
2480 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2486 if (!this->emitGetLocal(SubExprT, LocalIndex, E))
2496template <
class Emitter>
2503 auto visitChildExpr = [&](
const Expr *E) ->
bool {
2512 return visitChildExpr(TrueExpr);
2513 return visitChildExpr(FalseExpr);
2516 bool IsBcpCall =
false;
2517 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2518 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2522 LabelTy LabelEnd = this->getLabel();
2523 LabelTy LabelFalse = this->getLabel();
2526 if (!this->emitStartSpeculation(E))
2534 if (this->checkingForUndefinedBehavior()) {
2537 if (!this->
discard(FalseExpr))
2543 if (!this->jumpFalse(LabelFalse))
2545 if (!visitChildExpr(TrueExpr))
2547 if (!this->jump(LabelEnd))
2549 this->emitLabel(LabelFalse);
2550 if (!visitChildExpr(FalseExpr))
2552 this->fallthrough(LabelEnd);
2553 this->emitLabel(LabelEnd);
2556 return this->emitEndSpeculation(E);
2560template <
class Emitter>
2566 unsigned StringIndex =
P.createGlobalString(E);
2567 return this->emitGetPtrGlobal(StringIndex, E);
2572 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2573 assert(CAT &&
"a string literal that's not a constant array?");
2578 unsigned N = std::min(ArraySize, E->
getLength());
2581 for (
unsigned I = 0; I != N; ++I) {
2584 if (CharWidth == 1) {
2585 this->emitConstSint8(CodeUnit, E);
2586 this->emitInitElemSint8(I, E);
2587 }
else if (CharWidth == 2) {
2588 this->emitConstUint16(CodeUnit, E);
2589 this->emitInitElemUint16(I, E);
2590 }
else if (CharWidth == 4) {
2591 this->emitConstUint32(CodeUnit, E);
2592 this->emitInitElemUint32(I, E);
2594 llvm_unreachable(
"unsupported character width");
2599 for (
unsigned I = N; I != ArraySize; ++I) {
2600 if (CharWidth == 1) {
2601 this->emitConstSint8(0, E);
2602 this->emitInitElemSint8(I, E);
2603 }
else if (CharWidth == 2) {
2604 this->emitConstUint16(0, E);
2605 this->emitInitElemUint16(I, E);
2606 }
else if (CharWidth == 4) {
2607 this->emitConstUint32(0, E);
2608 this->emitInitElemUint32(I, E);
2610 llvm_unreachable(
"unsupported character width");
2617template <
class Emitter>
2621 return this->emitDummyPtr(E, E);
2624template <
class Emitter>
2626 auto &A =
Ctx.getASTContext();
2635template <
class Emitter>
2643 auto &A =
Ctx.getASTContext();
2647 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2648 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2655 unsigned StringIndex =
P.createGlobalString(SL);
2656 return this->emitGetPtrGlobal(StringIndex, E);
2659template <
class Emitter>
2663 return this->emitConst(E->
getValue(), E);
2666template <
class Emitter>
2692 if (!this->emitSetLocal(*RT, TempOffset, E))
2698 if (!this->emitLoad(LHST, E))
2702 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
2703 LHSComputationType, E))
2707 if (!this->emitGetLocal(*RT, TempOffset, E))
2712 if (!this->emitAddf(getFPOptions(E), E))
2716 if (!this->emitSubf(getFPOptions(E), E))
2720 if (!this->emitMulf(getFPOptions(E), E))
2724 if (!this->emitDivf(getFPOptions(E), E))
2735 return this->emitStorePop(LHST, E);
2736 return this->emitStore(LHST, E);
2739template <
class Emitter>
2748 if (Op != BO_AddAssign && Op != BO_SubAssign)
2757 if (!this->emitLoad(*
LT, LHS))
2763 if (Op == BO_AddAssign) {
2764 if (!this->emitAddOffset(*RT, E))
2767 if (!this->emitSubOffset(*RT, E))
2772 return this->emitStorePopPtr(E);
2773 return this->emitStorePtr(E);
2776template <
class Emitter>
2789 if (!
Ctx.getLangOpts().CPlusPlus14)
2790 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
2792 if (!
LT || !RT || !ResultT || !LHSComputationT)
2817 if (!this->emitSetLocal(*RT, TempOffset, E))
2824 if (!this->emitLoad(*
LT, E))
2826 if (
LT != LHSComputationT) {
2827 if (!this->emitCast(*
LT, *LHSComputationT, E))
2832 if (!this->emitGetLocal(*RT, TempOffset, E))
2838 if (!this->emitAdd(*LHSComputationT, E))
2842 if (!this->emitSub(*LHSComputationT, E))
2846 if (!this->emitMul(*LHSComputationT, E))
2850 if (!this->emitDiv(*LHSComputationT, E))
2854 if (!this->emitRem(*LHSComputationT, E))
2858 if (!this->emitShl(*LHSComputationT, *RT, E))
2862 if (!this->emitShr(*LHSComputationT, *RT, E))
2866 if (!this->emitBitAnd(*LHSComputationT, E))
2870 if (!this->emitBitXor(*LHSComputationT, E))
2874 if (!this->emitBitOr(*LHSComputationT, E))
2878 llvm_unreachable(
"Unimplemented compound assign operator");
2882 if (ResultT != LHSComputationT) {
2883 if (!this->emitCast(*LHSComputationT, *ResultT, E))
2890 return this->emitStoreBitFieldPop(*ResultT, E);
2891 return this->emitStorePop(*ResultT, E);
2894 return this->emitStoreBitField(*ResultT, E);
2895 return this->emitStore(*ResultT, E);
2898template <
class Emitter>
2906template <
class Emitter>
2918 return this->
discard(SubExpr);
2935 if (!this->
visit(SubExpr))
2937 if (!this->emitInitGlobalTemp(*SubExprT, *GlobalIndex, TempDecl, E))
2939 return this->emitGetPtrGlobal(*GlobalIndex, E);
2942 if (!this->checkLiteralType(SubExpr))
2945 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
2949 return this->emitInitGlobalTempComp(TempDecl, E);
2958 if (!this->
visit(SubExpr))
2960 if (!this->emitSetLocal(*SubExprT, LocalIndex, E))
2962 return this->emitGetPtrLocal(LocalIndex, E);
2965 if (!this->checkLiteralType(SubExpr))
2971 if (!this->emitGetPtrLocal(*LocalIndex, E))
2978template <
class Emitter>
2989 if (!this->
visit(SubExpr))
2993 return this->emitPopPtr(E);
2997template <
class Emitter>
3018 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3023 if (
P.isGlobalInitialized(*GlobalIndex))
3029 return this->emitInitGlobal(*
T, *GlobalIndex, E);
3041 unsigned LocalIndex;
3045 LocalIndex = *MaybeIndex;
3049 if (!this->emitGetPtrLocal(LocalIndex, E))
3053 return this->
visit(
Init) && this->emitInit(*
T, E);
3057template <
class Emitter>
3070template <
class Emitter>
3074 return this->emitConst(E->
getValue(), E);
3077template <
class Emitter>
3090 for (
const Record::Field &F : R->
fields()) {
3092 if (!
Init ||
Init->containsErrors())
3100 if (!this->emitInitField(*
T, F.Offset, E))
3103 if (!this->emitGetPtrField(F.Offset, E))
3109 if (!this->emitPopPtr(E))
3117template <
class Emitter>
3124 return this->emitGetPtrGlobal(StringIndex, E);
3130template <
class Emitter>
3135 return this->emitInvalid(E);
3138template <
class Emitter>
3164 if (PointeeToT && PointeeFromT) {
3180 bool Fatal = (ToT != FromT);
3187template <
class Emitter>
3190 if (!
Ctx.getLangOpts().CPlusPlus20) {
3198template <
class Emitter>
3204 return this->emitConstBool(E->
getValue(), E);
3207template <
class Emitter>
3212 if (
T->isRecordType()) {
3226 if (!this->emitGetPtrLocal(*LocalIndex, E))
3234 T->getAsCXXRecordDecl()))
3242 if (!this->visitZeroRecordInitializer(R, E))
3254 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3256 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3257 if (!this->emitCheckFunctionDecl(Ctor, E))
3268 assert(
Func->hasThisPointer());
3269 assert(!
Func->hasRVO());
3273 if (!this->emitDupPtr(E))
3277 for (
const auto *Arg : E->
arguments()) {
3278 if (!this->
visit(Arg))
3282 if (
Func->isVariadic()) {
3283 uint32_t VarArgSize = 0;
3284 unsigned NumParams =
Func->getNumWrittenParams();
3285 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3289 if (!this->emitCallVar(
Func, VarArgSize, E))
3292 if (!this->emitCall(
Func, 0, E)) {
3297 (void)this->emitPopPtr(E);
3303 return this->emitPopPtr(E);
3304 return this->emitFinishInit(E);
3307 if (
T->isArrayType()) {
3312 if (!this->emitDupPtr(E))
3316 initArrayDimension = [&](
QualType T) ->
bool {
3317 if (!
T->isArrayType()) {
3319 for (
const auto *Arg : E->
arguments()) {
3320 if (!this->
visit(Arg))
3324 return this->emitCall(
Func, 0, E);
3328 Ctx.getASTContext().getAsConstantArrayType(
T);
3333 for (
size_t I = 0; I != NumElems; ++I) {
3334 if (!this->emitConstUint64(I, E))
3336 if (!this->emitArrayElemPtrUint64(E))
3338 if (!initArrayDimension(ElemTy))
3341 return this->emitPopPtr(E);
3344 return initArrayDimension(E->
getType());
3350template <
class Emitter>
3360 assert(Val.
isInt());
3362 return this->emitConst(I, E);
3369 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3370 return this->
visit(LValueExpr);
3385 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3389 const APValue &
V = UGCD->getValue();
3390 for (
unsigned I = 0, N = R->
getNumFields(); I != N; ++I) {
3391 const Record::Field *F = R->
getField(I);
3392 const APValue &FieldValue =
V.getStructField(I);
3398 if (!this->emitInitField(FieldT, F->Offset, E))
3406template <
class Emitter>
3412 for (
unsigned I = 0; I != N; ++I) {
3419 if (!this->
discard(ArrayIndexExpr))
3424 if (!this->
visit(ArrayIndexExpr))
3428 if (!this->emitCast(IndexT,
PT_Sint64, E))
3438 return this->emitOffsetOf(
T, E, E);
3441template <
class Emitter>
3450 return this->visitZeroInitializer(*
T, Ty, E);
3457 if (!this->emitGetPtrLocal(*LocalIndex, E))
3462 QualType ElemQT = CT->getElementType();
3465 for (
unsigned I = 0; I != 2; ++I) {
3466 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3468 if (!this->emitInitElem(ElemT, I, E))
3480 if (!this->emitGetPtrLocal(*LocalIndex, E))
3485 QualType ElemQT = VT->getElementType();
3488 for (
unsigned I = 0, N = VT->getNumElements(); I != N; ++I) {
3489 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3491 if (!this->emitInitElem(ElemT, I, E))
3500template <
class Emitter>
3505template <
class Emitter>
3511template <
class Emitter>
3516template <
class Emitter>
3521 return this->emitConst(E->
getValue(), E);
3524template <
class Emitter>
3529 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3547 if (!this->emitGetParam(PT, Offset, E))
3552 return this->emitCall(F, 0, E);
3557template <
class Emitter>
3565 const Expr *PlacementDest =
nullptr;
3566 bool IsNoThrow =
false;
3568 if (PlacementArgs != 0) {
3577 if (PlacementArgs == 1) {
3585 if (!this->emitInvalidNewDeleteExpr(E, E))
3590 if (OperatorNew->isReservedGlobalPlacementOperator())
3591 PlacementDest = Arg1;
3595 return this->emitInvalid(E);
3597 }
else if (!OperatorNew
3598 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3599 return this->emitInvalidNewDeleteExpr(E, E);
3602 if (!PlacementDest) {
3607 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3610 Desc =
P.createDescriptor(
3613 false,
false,
false,
3619 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3623 const Expr *Stripped = *ArraySizeExpr;
3624 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3625 Stripped = ICE->getSubExpr())
3626 if (ICE->getCastKind() != CK_NoOp &&
3627 ICE->getCastKind() != CK_IntegralCast)
3635 if (!this->
visit(Stripped))
3637 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
3640 if (PlacementDest) {
3641 if (!this->
visit(PlacementDest))
3643 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3645 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
3648 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3653 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
3657 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
3664 size_t StaticInitElems = 0;
3665 const Expr *DynamicInit =
nullptr;
3667 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
3668 StaticInitElems = CAT->getZExtSize();
3672 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init);
3673 ILE && ILE->hasArrayFiller())
3674 DynamicInit = ILE->getArrayFiller();
3688 const Function *CtorFunc =
nullptr;
3689 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
3693 }
else if (!DynamicInit)
3696 LabelTy EndLabel = this->getLabel();
3697 LabelTy StartLabel = this->getLabel();
3702 if (!this->emitDupPtr(E))
3704 if (!this->emitNullPtr(0,
nullptr, E))
3706 if (!this->emitEQPtr(E))
3708 if (!this->jumpTrue(EndLabel))
3715 if (!this->emitConst(StaticInitElems,
SizeT, E))
3717 if (!this->emitSetLocal(
SizeT, Iter, E))
3720 this->fallthrough(StartLabel);
3721 this->emitLabel(StartLabel);
3723 if (!this->emitGetLocal(
SizeT, Iter, E))
3725 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
3727 if (!this->emitLT(
SizeT, E))
3729 if (!this->jumpFalse(EndLabel))
3733 if (!this->emitGetLocal(
SizeT, Iter, E))
3735 if (!this->emitArrayElemPtr(
SizeT, E))
3738 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
3743 if (!this->visitZeroInitializer(InitT, ElemType, E))
3745 if (!this->emitStorePop(InitT, E))
3747 }
else if (DynamicInit) {
3749 if (!this->
visit(DynamicInit))
3751 if (!this->emitStorePop(*InitT, E))
3756 if (!this->emitPopPtr(E))
3761 if (!this->emitCall(CtorFunc, 0, E))
3766 if (!this->emitGetPtrLocal(Iter, E))
3768 if (!this->emitIncPop(
SizeT,
false, E))
3771 if (!this->jump(StartLabel))
3774 this->fallthrough(EndLabel);
3775 this->emitLabel(EndLabel);
3779 if (PlacementDest) {
3780 if (!this->
visit(PlacementDest))
3782 if (!this->emitCheckNewTypeMismatch(E, E))
3787 if (!this->emitAlloc(Desc, E))
3796 if (!this->emitInit(*ElemT, E))
3807 return this->emitPopPtr(E);
3812template <
class Emitter>
3818 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3819 return this->emitInvalidNewDeleteExpr(E, E);
3822 if (!this->
visit(Arg))
3828template <
class Emitter>
3834 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
3839 return this->emitGetFnPtr(
Func, E);
3842template <
class Emitter>
3846 auto canonType = [](
const Type *
T) {
3847 return T->getCanonicalTypeUnqualified().getTypePtr();
3855 return this->emitGetTypeid(
3859 return this->emitGetTypeid(
3868 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
3874 if (!this->emitGetTypeidPtr(TypeInfoType, E))
3877 return this->emitPopPtr(E);
3881template <
class Emitter>
3883 assert(
Ctx.getLangOpts().CPlusPlus);
3884 return this->emitConstBool(E->
getValue(), E);
3887template <
class Emitter>
3899 return this->emitDummyPtr(GuidDecl, E);
3904 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3913 assert(
V.isStruct());
3914 assert(
V.getStructNumBases() == 0);
3918 return this->emitFinishInit(E);
3921template <
class Emitter>
3931template <
class Emitter>
3940template <
class Emitter>
3946template <
class Emitter>
3950 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
3954 if (OVE->isUnique())
3970template <
class Emitter>
3975template <
class Emitter>
3977 return this->emitError(E);
3980template <
class Emitter>
3986 return this->emitDummyPtr(E, E);
3989template <
class Emitter>
3993 QualType ElemType = VT->getElementType();
3997 PrimType SrcElemT = classifyVectorElementType(SrcType);
3999 unsigned SrcOffset =
4001 if (!this->
visit(Src))
4003 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4006 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4007 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4009 if (!this->emitArrayElemPop(SrcElemT, I, E))
4013 if (SrcElemT != ElemT) {
4014 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4016 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4017 const auto *TargetSemantics = &
Ctx.getFloatSemantics(ElemType);
4021 if (!this->emitInitElem(ElemT, I, E))
4028template <
class Emitter>
4032 return this->emitInvalid(E);
4042 assert(NumOutputElems > 0);
4045 unsigned VectorOffsets[2];
4046 for (
unsigned I = 0; I != 2; ++I) {
4049 if (!this->
visit(Vecs[I]))
4051 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4054 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4056 assert(ShuffleIndex >= -1);
4057 if (ShuffleIndex == -1)
4058 return this->emitInvalidShuffleVectorIndex(I, E);
4060 assert(ShuffleIndex < (NumInputElems * 2));
4061 if (!this->emitGetLocal(
PT_Ptr,
4062 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4064 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4065 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4068 if (!this->emitInitElem(ElemT, I, E))
4075template <
class Emitter>
4080 Base->getType()->isVectorType() ||
4086 if (Indices.size() == 1) {
4091 if (!this->emitConstUint32(Indices[0], E))
4093 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4103 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4111 if (!this->emitGetPtrLocal(*ResultIndex, E))
4119 uint32_t DstIndex = 0;
4120 for (uint32_t I : Indices) {
4121 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4123 if (!this->emitArrayElemPop(ElemT, I, E))
4125 if (!this->emitInitElem(ElemT, DstIndex, E))
4135template <
class Emitter>
4139 return this->
discard(SubExpr) && this->emitInvalid(E);
4145 return this->emitDummyPtr(E, E);
4148template <
class Emitter>
4153 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4158 if (!this->
visit(SubExpr))
4160 if (!this->emitConstUint8(0, E))
4162 if (!this->emitArrayElemPtrPopUint8(E))
4169 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4171 return this->emitInitField(SecondFieldT, R->
getField(1u)->
Offset, E);
4173 assert(SecondFieldT ==
PT_Ptr);
4177 if (!this->emitExpandPtr(E))
4181 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4186template <
class Emitter>
4201 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4203 return this->emitUnsupported(E);
4212 return this->
Visit(E);
4219 return this->
Visit(E);
4223 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4226 if (
const auto *CE = dyn_cast<CastExpr>(E);
4228 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4235 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4238 if (
const auto *CE = dyn_cast<CastExpr>(E);
4239 CE && (CE->getCastKind() == CK_DerivedToBase ||
4240 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4241 CE->getCastKind() == CK_NoOp))
4260 if (!this->emitGetPtrLocal(*LocalIndex, E))
4270 return this->
Visit(E);
4273template <
class Emitter>
4279 return this->
Visit(E);
4285 return this->
Visit(E);
4293 if (!this->
visit(E))
4295 return this->emitComplexBoolCast(E);
4300 if (!this->
visit(E))
4308 return this->emitIsNonNullPtr(E);
4312 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4315 return this->emitCast(*
T,
PT_Bool, E);
4318template <
class Emitter>
4322 QT = AT->getValueType();
4326 return this->emitZeroBool(E);
4328 return this->emitZeroSint8(E);
4330 return this->emitZeroUint8(E);
4332 return this->emitZeroSint16(E);
4334 return this->emitZeroUint16(E);
4336 return this->emitZeroSint32(E);
4338 return this->emitZeroUint32(E);
4340 return this->emitZeroSint64(E);
4342 return this->emitZeroUint64(E);
4344 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4346 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4348 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4351 return this->emitNullMemberPtr(0,
nullptr, E);
4353 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4354 return this->emitFloat(F, E);
4357 auto Sem = Ctx.getASTContext().getFixedPointSemantics(E->
getType());
4361 llvm_unreachable(
"unknown primitive type");
4364template <
class Emitter>
4365bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4370 for (
const Record::Field &Field : R->
fields()) {
4371 if (Field.isUnnamedBitField())
4378 if (!this->visitZeroInitializer(
T, QT, E))
4381 if (!this->emitInitFieldActivate(
T, Field.Offset, E))
4385 if (!this->emitInitField(
T, Field.Offset, E))
4390 if (!this->emitGetPtrField(Field.Offset, E))
4396 for (uint32_t I = 0, N = D->
getNumElems(); I != N; ++I) {
4397 if (!this->visitZeroInitializer(
T, ET, E))
4399 if (!this->emitInitElem(
T, I, E))
4404 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4407 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4415 if (!this->emitFinishInitActivatePop(E))
4419 if (!this->emitFinishInitPop(E))
4423 for (
const Record::Base &B : R->
bases()) {
4424 if (!this->emitGetPtrBase(B.Offset, E))
4426 if (!this->visitZeroRecordInitializer(B.R, E))
4428 if (!this->emitFinishInitPop(E))
4437template <
class Emitter>
4438bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4439 assert(
T->isArrayType() ||
T->isAnyComplexType() ||
T->isVectorType());
4440 const ArrayType *AT =
T->getAsArrayTypeUnsafe();
4445 for (
size_t I = 0; I != NumElems; ++I) {
4446 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4448 if (!this->emitInitElem(*ElemT, I, E))
4454 const Record *R = getRecord(ElemType);
4456 for (
size_t I = 0; I != NumElems; ++I) {
4457 if (!this->emitConstUint32(I, E))
4459 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4461 if (!this->visitZeroRecordInitializer(R, E))
4463 if (!this->emitPopPtr(E))
4469 for (
size_t I = 0; I != NumElems; ++I) {
4470 if (!this->emitConstUint32(I, E))
4472 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4474 if (!this->visitZeroArrayInitializer(ElemType, E))
4476 if (!this->emitPopPtr(E))
4485template <
class Emitter>
4486bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4488 if (!canClassify(E->
getType()))
4491 if (!this->visit(RHS))
4493 if (!this->visit(LHS))
4500 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4504 bool Activates = refersToUnion(LHS);
4507 if (!this->emitFlip(
PT_Ptr, RHT, E))
4510 if (DiscardResult) {
4511 if (BitField && Activates)
4512 return this->emitStoreBitFieldActivatePop(RHT, E);
4514 return this->emitStoreBitFieldPop(RHT, E);
4516 return this->emitStoreActivatePop(RHT, E);
4518 return this->emitStorePop(RHT, E);
4521 auto maybeLoad = [&](
bool Result) ->
bool {
4527 return this->emitLoadPop(RHT, E);
4531 if (BitField && Activates)
4532 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4534 return maybeLoad(this->emitStoreBitField(RHT, E));
4536 return maybeLoad(this->emitStoreActivate(RHT, E));
4538 return maybeLoad(this->emitStore(RHT, E));
4541template <
class Emitter>
4542template <
typename T>
4546 return this->emitConstSint8(
Value, E);
4548 return this->emitConstUint8(
Value, E);
4550 return this->emitConstSint16(
Value, E);
4552 return this->emitConstUint16(
Value, E);
4554 return this->emitConstSint32(
Value, E);
4556 return this->emitConstUint32(
Value, E);
4558 return this->emitConstSint64(
Value, E);
4560 return this->emitConstUint64(
Value, E);
4562 return this->emitConstBool(
Value, E);
4569 llvm_unreachable(
"Invalid integral type");
4572 llvm_unreachable(
"unknown primitive type");
4575template <
class Emitter>
4576template <
typename T>
4577bool Compiler<Emitter>::emitConst(
T Value,
const Expr *E) {
4578 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
4581template <
class Emitter>
4585 return this->emitConstIntAPS(
Value, E);
4587 return this->emitConstIntAP(
Value, E);
4589 if (
Value.isSigned())
4590 return this->emitConst(
Value.getSExtValue(), Ty, E);
4591 return this->emitConst(
Value.getZExtValue(), Ty, E);
4594template <
class Emitter>
4598 return this->emitConstIntAPS(
Value, E);
4600 return this->emitConstIntAP(
Value, E);
4603 return this->emitConst(
Value.getSExtValue(), Ty, E);
4604 return this->emitConst(
Value.getZExtValue(), Ty, E);
4607template <
class Emitter>
4608bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
4609 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
4612template <
class Emitter>
4624 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
4625 Locals.insert({VD, Local});
4627 VarScope->addExtended(Local, ExtendingDecl);
4629 VarScope->addForScopeKind(Local, SC);
4630 return Local.Offset;
4633template <
class Emitter>
4637 bool IsConstexprUnknown) {
4640 bool IsTemporary =
false;
4641 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
4644 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
4645 Init = VarD->getInit();
4647 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
4655 IsTemporary,
false,
false,
Init);
4657 return std::nullopt;
4662 Locals.insert({Key, Local});
4664 VarScope->addExtended(Local, ExtendingDecl);
4666 VarScope->addForScopeKind(Local, SC);
4667 return Local.Offset;
4670template <
class Emitter>
4680 return std::nullopt;
4690 return Local.Offset;
4693template <
class Emitter>
4695 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
4696 return PT->getPointeeType()->getAsCanonical<RecordType>();
4702 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
4706template <
class Emitter>
4708 return P.getOrCreateRecord(RD);
4711template <
class Emitter>
4713 return Ctx.getOrCreateFunction(FD);
4716template <
class Emitter>
4721 if (!DestroyToplevelScope) {
4722 if (!this->emitCheckAllocations(E))
4726 auto maybeDestroyLocals = [&]() ->
bool {
4727 if (DestroyToplevelScope)
4728 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
4729 return this->emitCheckAllocations(E);
4736 return this->emitRetVoid(E) && maybeDestroyLocals();
4744 return this->emitRet(*
T, E) && maybeDestroyLocals();
4752 if (!this->emitGetPtrLocal(*LocalOffset, E))
4758 if (!this->emitFinishInit(E))
4763 return this->emitRetValue(E) && maybeDestroyLocals();
4766 return maybeDestroyLocals() && this->emitCheckAllocations(E) &&
false;
4769template <
class Emitter>
4771 bool IsConstexprUnknown) {
4774 IsConstexprUnknown);
4783 if (
auto GlobalIndex =
P.getGlobal(VD)) {
4784 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
4799template <
class Emitter>
4801 bool ConstantContext) {
4804 if (!ConstantContext) {
4818 auto GlobalIndex =
P.getGlobal(VD);
4819 assert(GlobalIndex);
4821 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
4824 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
4828 auto Local =
Locals.find(VD);
4829 assert(Local !=
Locals.end());
4831 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
4834 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
4844 auto GlobalIndex =
P.getGlobal(VD);
4845 assert(GlobalIndex);
4846 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
4856 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
4859template <
class Emitter>
4862 bool Toplevel,
bool IsConstexprUnknown) {
4869 if (!this->isActive())
4874 if (
Init &&
Init->isValueDependent())
4878 auto checkDecl = [&]() ->
bool {
4880 return !NeedsOp || this->emitCheckDecl(VD, VD);
4888 if (
P.getPtrGlobal(*GlobalIndex).isInitialized())
4892 }
else if ((GlobalIndex =
P.createGlobal(VD,
Init))) {
4906 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
4909 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
4915 return this->emitFinishInitGlobal(
Init);
4924 IsConstexprUnknown);
4935 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
4939 return this->emitSetLocal(*VarT, Offset, VD);
4947 if (!this->emitGetPtrLocal(*Offset,
Init))
4953 return this->emitFinishInitPop(
Init);
4958template <
class Emitter>
4963 return this->emitConst(Val.
getInt(), ValType, E);
4966 return this->emitFloat(F, E);
4971 return this->emitNull(ValType, 0,
nullptr, E);
4973 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
4974 return this->
visit(BaseExpr);
4979 return this->emitGetMemberPtr(MemberDecl, E);
4980 return this->emitNullMemberPtr(0,
nullptr, E);
4986template <
class Emitter>
4994 const Record::Field *RF = R->
getField(I);
4995 QualType FieldType = RF->Decl->getType();
5000 if (!this->emitInitField(*PT, RF->Offset, E))
5003 if (!this->emitGetPtrField(RF->Offset, E))
5007 if (!this->emitPopPtr(E))
5018 const Record::Field *RF = R->
getField(UnionField);
5022 return this->emitInitField(
T, RF->Offset, E);
5025 const auto *ArrType =
T->getAsArrayTypeUnsafe();
5026 QualType ElemType = ArrType->getElementType();
5027 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5032 if (!this->emitInitElem(*ElemT, A, E))
5035 if (!this->emitConstUint32(A, E))
5037 if (!this->emitArrayElemPtrUint32(E))
5041 if (!this->emitPopPtr(E))
5052template <
class Emitter>
5054 unsigned BuiltinID) {
5055 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5060 return this->emitConst(0, E);
5063 if (!this->emitStartSpeculation(E))
5065 LabelTy EndLabel = this->getLabel();
5066 if (!this->speculate(E, EndLabel))
5068 this->fallthrough(EndLabel);
5069 if (!this->emitEndSpeculation(E))
5078 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5079 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5080 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5081 BuiltinID == Builtin::BI__builtin_function_start) {
5084 return this->emitDummyPtr(E, E);
5095 if (!this->emitGetPtrLocal(*LocalIndex, E))
5100 switch (BuiltinID) {
5101 case Builtin::BI__builtin_object_size:
5102 case Builtin::BI__builtin_dynamic_object_size: {
5106 if (!this->
visit(Arg0))
5120 for (
const auto *Arg : E->
arguments()) {
5121 if (!this->
visit(Arg))
5127 if (!this->emitCallBI(E, BuiltinID, E))
5132 return this->emitPop(*ReturnT, E);
5138template <
class Emitter>
5155 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5156 DD && DD->isTrivial()) {
5158 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5160 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5161 this->emitPopPtr(E);
5177 if (!this->emitGetPtrLocal(*LocalIndex, E))
5185 if (!this->emitGetPtrLocal(*LocalIndex, E))
5189 if (!this->emitDupPtr(E))
5196 bool IsAssignmentOperatorCall =
false;
5197 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5198 OCE && OCE->isAssignmentOp()) {
5202 assert(Args.size() == 2);
5203 IsAssignmentOperatorCall =
true;
5204 std::reverse(Args.begin(), Args.end());
5210 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5211 MD && MD->isStatic()) {
5215 Args.erase(Args.begin());
5219 bool Devirtualized =
false;
5222 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5230 if (!this->
visit(Callee))
5232 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5234 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5236 if (!this->emitGetMemberPtrBase(E))
5239 const auto *InstancePtr = MC->getImplicitObjectArgument();
5246 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5247 Devirtualized =
true;
5248 if (!this->
visit(Stripped))
5251 if (!this->
visit(InstancePtr))
5255 if (!this->
visit(InstancePtr))
5259 }
else if (
const auto *PD =
5260 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5261 if (!this->emitCheckPseudoDtor(E))
5269 return this->emitEndLifetimePop(E);
5270 }
else if (!FuncDecl) {
5274 if (!this->
visit(Callee))
5276 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5280 if (!this->
visitCallArgs(Args, FuncDecl, IsAssignmentOperatorCall,
5285 if (IsAssignmentOperatorCall) {
5286 assert(Args.size() == 2);
5289 if (!this->emitFlip(Arg2T, Arg1T, E))
5303 assert(HasRVO ==
Func->hasRVO());
5305 bool HasQualifier =
false;
5306 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5307 HasQualifier = ME->hasQualifier();
5309 bool IsVirtual =
false;
5310 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5311 IsVirtual = !Devirtualized && MD->isVirtual();
5316 if (IsVirtual && !HasQualifier) {
5317 uint32_t VarArgSize = 0;
5318 unsigned NumParams =
5320 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5323 if (!this->emitCallVirt(
Func, VarArgSize, E))
5325 }
else if (
Func->isVariadic()) {
5326 uint32_t VarArgSize = 0;
5327 unsigned NumParams =
5329 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5331 if (!this->emitCallVar(
Func, VarArgSize, E))
5334 if (!this->emitCall(
Func, 0, E))
5343 uint32_t ArgSize = 0;
5344 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5350 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5352 if (!this->emitGetMemberPtrDecl(E))
5355 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5358 if (!this->emitCallPtr(ArgSize, E, E))
5369template <
class Emitter>
5376template <
class Emitter>
5383template <
class Emitter>
5388 return this->emitConstBool(E->
getValue(), E);
5391template <
class Emitter>
5397 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5398 return this->emitNullPtr(Val,
nullptr, E);
5401template <
class Emitter>
5409 return this->emitZero(
T, E);
5412template <
class Emitter>
5417 if (this->LambdaThisCapture.Offset > 0) {
5418 if (this->LambdaThisCapture.IsPtr)
5419 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
5420 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
5428 return this->emitThis(E);
5443 unsigned StartIndex = 0;
5444 unsigned EndIndex = 0;
5446 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
5448 EndIndex = StartIndex;
5455 for (; StartIndex > 0; --StartIndex) {
5465 if (StartIndex == 0 && EndIndex == 0)
5468 assert(StartIndex < EndIndex);
5471 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
5483 case Stmt::CompoundStmtClass:
5485 case Stmt::DeclStmtClass:
5487 case Stmt::ReturnStmtClass:
5489 case Stmt::IfStmtClass:
5491 case Stmt::WhileStmtClass:
5493 case Stmt::DoStmtClass:
5495 case Stmt::ForStmtClass:
5497 case Stmt::CXXForRangeStmtClass:
5499 case Stmt::BreakStmtClass:
5501 case Stmt::ContinueStmtClass:
5503 case Stmt::SwitchStmtClass:
5505 case Stmt::CaseStmtClass:
5507 case Stmt::DefaultStmtClass:
5509 case Stmt::AttributedStmtClass:
5511 case Stmt::CXXTryStmtClass:
5513 case Stmt::NullStmtClass:
5516 case Stmt::GCCAsmStmtClass:
5517 case Stmt::MSAsmStmtClass:
5518 case Stmt::GotoStmtClass:
5519 return this->emitInvalid(S);
5520 case Stmt::LabelStmtClass:
5523 if (
const auto *E = dyn_cast<Expr>(S))
5530template <
class Emitter>
5533 for (
const auto *InnerStmt : S->
body())
5536 return Scope.destroyLocals();
5539template <
class Emitter>
5540bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
5541 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
5542 for (
auto *BD : DD->flat_bindings())
5543 if (
auto *KD = BD->getHoldingVar();
5544 KD && !this->visitVarDecl(KD, KD->getInit()))
5553 const auto *CXXRD = dyn_cast<CXXRecordDecl>(FD->
getParent());
5554 return !CXXRD || CXXRD->hasTrivialDefaultConstructor();
5557template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
5559 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
5560 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
5567 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
5572 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
5573 ICE && (ICE->getCastKind() == CK_NoOp ||
5574 ICE->getCastKind() == CK_DerivedToBase ||
5575 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
5576 E = ICE->getSubExpr();
5580 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
5581 const auto *ThisRecord =
5582 This->getType()->getPointeeType()->getAsRecordDecl();
5583 if (!ThisRecord->isUnion())
5586 if (
const auto *Ctor =
5587 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
5588 return Ctor->getParent() == ThisRecord;
5597template <
class Emitter>
5599 bool EvaluateConditionDecl) {
5600 for (
const auto *D : DS->
decls()) {
5605 const auto *VD = dyn_cast<VarDecl>(D);
5612 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
5619template <
class Emitter>
5622 return this->emitUnsupported(RS);
5628 if (!this->
visit(RE))
5634 if (RE->getType()->isVoidType()) {
5635 if (!this->
visit(RE))
5640 if (!this->emitRVOPtr(RE))
5644 if (!this->emitPopPtr(RE))
5648 return this->emitRetVoid(RS);
5654 return this->emitRetVoid(RS);
5658 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
5664 if (
auto *CondInit = IS->
getInit())
5678 return visitChildStmt(IS->
getThen());
5680 return visitChildStmt(Else);
5686 if (!this->emitIsConstantContext(IS))
5689 if (!this->emitIsConstantContext(IS))
5691 if (!this->emitInv(IS))
5702 LabelTy LabelElse = this->getLabel();
5703 LabelTy LabelEnd = this->getLabel();
5704 if (!this->jumpFalse(LabelElse))
5706 if (!visitChildStmt(IS->
getThen()))
5708 if (!this->jump(LabelEnd))
5710 this->emitLabel(LabelElse);
5711 if (!visitChildStmt(Else))
5713 this->emitLabel(LabelEnd);
5715 LabelTy LabelEnd = this->getLabel();
5716 if (!this->jumpFalse(LabelEnd))
5718 if (!visitChildStmt(IS->
getThen()))
5720 this->emitLabel(LabelEnd);
5726template <
class Emitter>
5731 LabelTy CondLabel = this->getLabel();
5732 LabelTy EndLabel = this->getLabel();
5736 this->fallthrough(CondLabel);
5737 this->emitLabel(CondLabel);
5751 if (!this->jumpFalse(EndLabel))
5760 if (!this->jump(CondLabel))
5762 this->fallthrough(EndLabel);
5763 this->emitLabel(EndLabel);
5771 LabelTy StartLabel = this->getLabel();
5772 LabelTy EndLabel = this->getLabel();
5773 LabelTy CondLabel = this->getLabel();
5777 this->fallthrough(StartLabel);
5778 this->emitLabel(StartLabel);
5784 this->fallthrough(CondLabel);
5785 this->emitLabel(CondLabel);
5792 if (!this->jumpTrue(StartLabel))
5795 this->fallthrough(EndLabel);
5796 this->emitLabel(EndLabel);
5800template <
class Emitter>
5808 LabelTy EndLabel = this->getLabel();
5809 LabelTy CondLabel = this->getLabel();
5810 LabelTy IncLabel = this->getLabel();
5817 this->fallthrough(CondLabel);
5818 this->emitLabel(CondLabel);
5830 if (!this->jumpFalse(EndLabel))
5839 this->fallthrough(IncLabel);
5840 this->emitLabel(IncLabel);
5846 if (!this->jump(CondLabel))
5850 this->emitLabel(EndLabel);
5856template <
class Emitter>
5866 LabelTy EndLabel = this->getLabel();
5867 LabelTy CondLabel = this->getLabel();
5868 LabelTy IncLabel = this->getLabel();
5883 this->fallthrough(CondLabel);
5884 this->emitLabel(CondLabel);
5887 if (!this->jumpFalse(EndLabel))
5898 this->fallthrough(IncLabel);
5899 this->emitLabel(IncLabel);
5904 if (!this->jump(CondLabel))
5907 this->fallthrough(EndLabel);
5908 this->emitLabel(EndLabel);
5912template <
class Emitter>
5923 if (LI.BreakLabel) {
5924 TargetLabel = *LI.BreakLabel;
5925 BreakScope = LI.BreakOrContinueScope;
5931 if (LI.Name == TargetLoop) {
5932 TargetLabel = *LI.BreakLabel;
5933 BreakScope = LI.BreakOrContinueScope;
5939 assert(TargetLabel);
5943 C->emitDestruction();
5945 return this->jump(*TargetLabel);
5948template <
class Emitter>
5959 if (LI.ContinueLabel) {
5960 TargetLabel = *LI.ContinueLabel;
5961 ContinueScope = LI.BreakOrContinueScope;
5967 if (LI.Name == TargetLoop) {
5968 TargetLabel = *LI.ContinueLabel;
5969 ContinueScope = LI.BreakOrContinueScope;
5974 assert(TargetLabel);
5978 C->emitDestruction();
5980 return this->jump(*TargetLabel);
5983template <
class Emitter>
5986 if (
Cond->containsErrors())
5992 LabelTy EndLabel = this->getLabel();
5997 if (
const auto *CondInit = S->
getInit())
6008 if (!this->emitSetLocal(CondT, CondVar, S))
6018 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6020 if (CS->caseStmtIsGNURange())
6025 if (
Value->isValueDependent())
6030 if (!this->emitGetLocal(CondT, CondVar, CS))
6036 if (!this->emitEQ(ValueT, S))
6041 assert(!DefaultLabel);
6042 DefaultLabel = this->getLabel();
6049 if (!this->jump(*DefaultLabel))
6052 if (!this->jump(EndLabel))
6060 this->emitLabel(EndLabel);
6065template <
class Emitter>
6071template <
class Emitter>
6078 if (LI.DefaultLabel) {
6079 DefaultLabel = *LI.DefaultLabel;
6084 this->emitLabel(DefaultLabel);
6088template <
class Emitter>
6090 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6091 !this->Ctx.getLangOpts().MSVCCompat) {
6093 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6099 const Expr *Assumption = AA->getAssumption();
6110 if (!this->emitAssume(Assumption))
6119template <
class Emitter>
6125template <
class Emitter>
6126bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6133 assert(ClosureClass->
captures().empty());
6137 "A generic lambda's static-invoker function must be a "
6138 "template specialization");
6142 void *InsertPos =
nullptr;
6143 const FunctionDecl *CorrespondingCallOpSpecialization =
6145 assert(CorrespondingCallOpSpecialization);
6146 LambdaCallOp = CorrespondingCallOpSpecialization;
6150 assert(ClosureClass->
captures().empty());
6151 const Function *
Func = this->getFunction(LambdaCallOp);
6154 assert(
Func->hasThisPointer());
6157 if (
Func->hasRVO()) {
6158 if (!this->emitRVOPtr(MD))
6166 if (!this->emitNullPtr(0,
nullptr, MD))
6171 auto It = this->Params.find(PVD);
6172 assert(It != this->Params.end());
6176 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6177 if (!this->emitGetParam(ParamType, It->second.Offset, MD))
6181 if (!this->emitCall(
Func, 0, LambdaCallOp))
6186 return this->emitRet(*ReturnType, MD);
6189 return this->emitRetVoid(MD);
6192template <
class Emitter>
6193bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6194 if (Ctx.getLangOpts().CPlusPlus23)
6204 const Expr *InitExpr =
Init->getInit();
6206 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6210 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6220template <
class Emitter>
6222 assert(!ReturnType);
6224 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6225 const Expr *InitExpr,
6228 if (InitExpr->getType().isNull())
6232 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6235 if (!this->visit(InitExpr))
6238 bool BitField = F->isBitField();
6240 return this->emitInitThisBitField(*
T, F, FieldOffset, InitExpr);
6241 return this->emitInitThisField(*
T, FieldOffset, InitExpr);
6246 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6249 if (
Activate && !this->emitActivate(InitExpr))
6252 if (!this->visitInitializer(InitExpr))
6255 return this->emitFinishInitPop(InitExpr);
6259 const Record *R = this->getRecord(RD);
6268 return this->emitRetVoid(Ctor);
6271 if (!this->emitThis(Ctor))
6280 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6281 this->emitRetVoid(Ctor);
6285 for (
const auto *
Init : Ctor->
inits()) {
6287 LocalScope<Emitter>
Scope(
this);
6289 const Expr *InitExpr =
Init->getInit();
6295 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6298 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6301 if (
Init->isBaseVirtual()) {
6303 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6309 const Record::Base *B = R->
getBase(BaseDecl);
6311 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6315 if (IsUnion && !this->emitActivate(InitExpr))
6318 if (!this->visitInitializer(InitExpr))
6320 if (!this->emitFinishInitPop(InitExpr))
6325 assert(IFD->getChainingSize() >= 2);
6327 unsigned NestedFieldOffset = 0;
6328 const Record::Field *NestedField =
nullptr;
6329 for (
const NamedDecl *ND : IFD->chain()) {
6331 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6332 assert(FieldRecord);
6334 NestedField = FieldRecord->
getField(FD);
6335 assert(NestedField);
6336 IsUnion = IsUnion || FieldRecord->
isUnion();
6338 NestedFieldOffset += NestedField->Offset;
6340 assert(NestedField);
6342 unsigned FirstLinkOffset =
6346 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
6351 unsigned InitFieldOffset = 0;
6352 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
6354 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6355 assert(FieldRecord);
6356 NestedField = FieldRecord->
getField(FD);
6357 InitFieldOffset += NestedField->Offset;
6358 assert(NestedField);
6359 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
6361 if (!this->emitFinishInitPop(InitExpr))
6366 assert(
Init->isDelegatingInitializer());
6367 if (!this->emitThis(InitExpr))
6369 if (!this->visitInitializer(
Init->getInit()))
6371 if (!this->emitPopPtr(InitExpr))
6375 if (!
Scope.destroyLocals())
6379 if (
const auto *Body = Ctor->
getBody())
6380 if (!visitStmt(Body))
6386template <
class Emitter>
6389 const Record *R = this->getRecord(RD);
6394 if (!this->visitStmt(Dtor->
getBody()))
6398 if (!this->emitThis(Dtor))
6401 if (!this->emitCheckDestruction(Dtor))
6409 for (
const Record::Field &Field : llvm::reverse(R->
fields())) {
6413 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
6415 if (!this->emitDestructionPop(D,
SourceInfo{}))
6420 for (
const Record::Base &
Base : llvm::reverse(R->
bases())) {
6421 if (
Base.R->hasTrivialDtor())
6425 if (!this->emitRecordDestructionPop(
Base.R, {}))
6430 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
6433template <
class Emitter>
6434bool Compiler<Emitter>::compileUnionAssignmentOperator(
6436 if (!this->emitThis(MD))
6445 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
6448template <
class Emitter>
6455 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
6456 return this->compileConstructor(Ctor);
6457 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
6458 return this->compileDestructor(Dtor);
6461 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
6466 return this->compileUnionAssignmentOperator(MD);
6469 return this->emitLambdaStaticInvokerBody(MD);
6473 if (
const auto *Body = F->
getBody())
6487 return FD->getBitWidthValue();
6490template <
class Emitter>
6503 if (!
Ctx.getLangOpts().CPlusPlus14)
6504 return this->emitInvalid(E);
6506 return this->emitError(E);
6508 if (!this->
visit(SubExpr))
6512 if (!this->emitIncPtr(E))
6519 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
6520 : this->emitIncf(getFPOptions(E), E);
6532 if (!
Ctx.getLangOpts().CPlusPlus14)
6533 return this->emitInvalid(E);
6535 return this->emitError(E);
6537 if (!this->
visit(SubExpr))
6541 if (!this->emitDecPtr(E))
6548 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
6549 : this->emitDecf(getFPOptions(E), E);
6562 if (!
Ctx.getLangOpts().CPlusPlus14)
6563 return this->emitInvalid(E);
6565 return this->emitError(E);
6567 if (!this->
visit(SubExpr))
6571 if (!this->emitLoadPtr(E))
6573 if (!this->emitConstUint8(1, E))
6575 if (!this->emitAddOffsetUint8(E))
6577 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
6583 return this->emitIncfPop(getFPOptions(E), E);
6593 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
6594 if (!this->emitLoadFloat(E))
6596 APFloat F(TargetSemantics, 1);
6597 if (!this->emitFloat(F, E))
6600 if (!this->emitAddf(getFPOptions(E), E))
6602 if (!this->emitStoreFloat(E))
6614 return E->
isGLValue() || this->emitLoadPop(*
T, E);
6617 if (!
Ctx.getLangOpts().CPlusPlus14)
6618 return this->emitInvalid(E);
6620 return this->emitError(E);
6622 if (!this->
visit(SubExpr))
6626 if (!this->emitLoadPtr(E))
6628 if (!this->emitConstUint8(1, E))
6630 if (!this->emitSubOffsetUint8(E))
6632 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
6638 return this->emitDecfPop(getFPOptions(E), E);
6648 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
6649 if (!this->emitLoadFloat(E))
6651 APFloat F(TargetSemantics, 1);
6652 if (!this->emitFloat(F, E))
6655 if (!this->emitSubf(getFPOptions(E), E))
6657 if (!this->emitStoreFloat(E))
6669 return E->
isGLValue() || this->emitLoadPop(*
T, E);
6673 return this->emitError(E);
6676 return this->
discard(SubExpr);
6681 if (!this->emitInv(E))
6685 return this->emitCast(
PT_Bool, ET, E);
6689 return this->emitError(E);
6691 if (!this->
visit(SubExpr))
6696 return this->emitError(E);
6698 if (!this->
visit(SubExpr))
6711 return this->
discard(SubExpr);
6713 if (!this->
visit(SubExpr))
6720 return this->emitNarrowPtr(E);
6725 return this->emitError(E);
6727 if (!this->
visit(SubExpr))
6737 return this->visitZeroInitializer(*
T, SubExpr->
getType(), SubExpr);
6742 assert(
false &&
"Unhandled opcode");
6748template <
class Emitter>
6754 return this->
discard(SubExpr);
6757 auto prepareResult = [=]() ->
bool {
6762 return this->emitGetPtrLocal(*LocalIndex, E);
6769 unsigned SubExprOffset = ~0u;
6770 auto createTemp = [=, &SubExprOffset]() ->
bool {
6773 if (!this->
visit(SubExpr))
6775 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
6779 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
6780 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
6782 return this->emitArrayElemPop(ElemT, Index, E);
6787 if (!prepareResult())
6791 for (
unsigned I = 0; I != 2; ++I) {
6792 if (!getElem(SubExprOffset, I))
6794 if (!this->emitNeg(ElemT, E))
6796 if (!this->emitInitElem(ElemT, I, E))
6807 if (!this->
visit(SubExpr))
6809 if (!this->emitComplexBoolCast(SubExpr))
6811 if (!this->emitInv(E))
6814 return this->emitCast(
PT_Bool, ET, E);
6818 return this->emitComplexReal(SubExpr);
6821 if (!this->
visit(SubExpr))
6825 if (!this->emitConstUint8(1, E))
6827 return this->emitArrayElemPtrPopUint8(E);
6835 if (!this->
visit(SubExpr))
6838 if (!this->emitArrayElem(ElemT, 1, E))
6840 if (!this->emitNeg(ElemT, E))
6842 if (!this->emitInitElem(ElemT, 1, E))
6850 return this->emitInvalid(E);
6856template <
class Emitter>
6862 return this->
discard(SubExpr);
6865 if (UnaryOp == UO_Extension)
6868 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
6869 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
6870 return this->emitInvalid(E);
6873 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
6880 if (!this->emitGetPtrLocal(*LocalIndex, E))
6885 unsigned SubExprOffset =
6887 if (!this->
visit(SubExpr))
6889 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
6894 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
6895 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
6897 return this->emitArrayElemPop(ElemT, Index, E);
6902 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6903 if (!getElem(SubExprOffset, I))
6905 if (!this->emitNeg(ElemT, E))
6907 if (!this->emitInitElem(ElemT, I, E))
6922 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6923 if (!getElem(SubExprOffset, I))
6926 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
6928 if (!this->emitInv(E))
6930 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
6932 if (!this->emitNeg(ElemT, E))
6934 if (ElemT != ResultVecElemT &&
6935 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
6937 if (!this->emitInitElem(ResultVecElemT, I, E))
6943 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6944 if (!getElem(SubExprOffset, I))
6947 if (!this->emitInv(E))
6950 if (!this->emitComp(ElemT, E))
6953 if (!this->emitInitElem(ElemT, I, E))
6958 llvm_unreachable(
"Unsupported unary operators should be handled up front");
6963template <
class Emitter>
6968 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D))
6969 return this->emitConst(ECD->getInitVal(), E);
6970 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
6972 return F && this->emitGetFnPtr(F, E);
6974 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
6976 if (!this->emitGetPtrGlobal(*Index, E))
6981 return this->emitInitGlobal(*
T, *Index, E);
6998 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
6999 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7004 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7005 if (IsReference || !It->second.IsPtr)
7006 return this->emitGetParam(
classifyPrim(E), It->second.Offset, E);
7008 return this->emitGetPtrParam(It->second.Offset, E);
7013 const unsigned Offset = It->second.Offset;
7016 return this->emitGetPtrLocal(Offset, E);
7019 if (
auto GlobalIndex =
P.getGlobal(D)) {
7021 if (!
Ctx.getLangOpts().CPlusPlus11)
7022 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7023 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7026 return this->emitGetPtrGlobal(*GlobalIndex, E);
7030 auto revisit = [&](
const VarDecl *VD) ->
bool {
7033 auto VarState = this->
visitDecl(VD,
true);
7035 if (!this->emitPopCC(E))
7038 if (VarState.notCreated())
7047 if (
auto It = this->LambdaCaptures.find(D);
7048 It != this->LambdaCaptures.end()) {
7049 auto [Offset, IsPtr] = It->second;
7052 return this->emitGetThisFieldPtr(Offset, E);
7053 return this->emitGetPtrThisField(Offset, E);
7056 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7057 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7058 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7062 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7063 return this->
visit(BD->getBinding());
7067 return this->emitDummyPtr(D, E);
7072 if (!
Ctx.getLangOpts().CPlusPlus) {
7073 if (
const auto *VD = dyn_cast<VarDecl>(D);
7077 return this->emitDummyPtr(D, E);
7081 const auto *VD = dyn_cast<VarDecl>(D);
7083 return this->emitDummyPtr(D, E);
7085 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7086 if (
T.isConstant(
Ctx.getASTContext()))
7088 return T->isReferenceType();
7092 typeShouldBeVisited(VD->
getType())) {
7094 Init && !
Init->isValueDependent()) {
7100 (void)
Init->EvaluateAsInitializer(
V,
Ctx.getASTContext(), VD, Notes,
7118 return this->emitDummyPtr(D, E);
7124 return this->emitDummyPtr(D, E);
7127template <
class Emitter>
7135 C->emitDestruction();
7138template <
class Emitter>
7139unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7142 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7144 return Ty->getAsCXXRecordDecl();
7146 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7147 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7149 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7153template <
class Emitter>
7160 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7165 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7166 getFPOptions(E), E);
7168 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7169 getFPOptions(E), E);
7173 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7178 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7180 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7184 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7188 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7189 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7197template <
class Emitter>
7198bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7202 return this->
discard(SubExpr);
7204 if (!this->visit(SubExpr))
7207 if (!this->emitConstUint8(0, SubExpr))
7209 return this->emitArrayElemPtrPopUint8(SubExpr);
7213 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7217template <
class Emitter>
7218bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7219 assert(!DiscardResult);
7223 if (!this->emitArrayElem(ElemT, 0, E))
7226 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7229 if (!this->emitCast(ElemT,
PT_Bool, E))
7234 LabelTy LabelTrue = this->getLabel();
7235 if (!this->jumpTrue(LabelTrue))
7238 if (!this->emitArrayElemPop(ElemT, 1, E))
7241 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7244 if (!this->emitCast(ElemT,
PT_Bool, E))
7248 LabelTy EndLabel = this->getLabel();
7249 this->jump(EndLabel);
7251 this->emitLabel(LabelTrue);
7252 if (!this->emitPopPtr(E))
7254 if (!this->emitConstBool(
true, E))
7257 this->fallthrough(EndLabel);
7258 this->emitLabel(EndLabel);
7263template <
class Emitter>
7264bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
7268 assert(!DiscardResult);
7274 LHSIsComplex =
true;
7275 ElemT = classifyComplexElementType(LHS->
getType());
7276 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
7277 if (!this->visit(LHS))
7279 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
7282 LHSIsComplex =
false;
7284 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
7285 if (!this->visit(LHS))
7287 if (!this->emitSetLocal(LHST, LHSOffset, E))
7294 RHSIsComplex =
true;
7295 ElemT = classifyComplexElementType(RHS->
getType());
7296 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
7297 if (!this->visit(RHS))
7299 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
7302 RHSIsComplex =
false;
7304 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
7305 if (!this->visit(RHS))
7307 if (!this->emitSetLocal(RHST, RHSOffset, E))
7311 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
7312 bool IsComplex) ->
bool {
7314 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
7316 return this->emitArrayElemPop(ElemT, Index, E);
7318 return this->emitGetLocal(ElemT, LocalOffset, E);
7321 for (
unsigned I = 0; I != 2; ++I) {
7323 if (!getElem(LHSOffset, I, LHSIsComplex))
7325 if (!getElem(RHSOffset, I, RHSIsComplex))
7328 if (!this->emitEQ(ElemT, E))
7331 if (!this->emitCastBoolUint8(E))
7336 if (!this->emitAddUint8(E))
7338 if (!this->emitConstUint8(2, E))
7342 if (!this->emitEQUint8(E))
7345 if (!this->emitNEUint8(E))
7352 return this->emitCast(
PT_Bool, ResT, E);
7359template <
class Emitter>
7360bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
7366 const Function *DtorFunc = getFunction(Dtor);
7369 assert(DtorFunc->hasThisPointer());
7370 assert(DtorFunc->getNumParams() == 1);
7371 return this->emitCall(DtorFunc, 0, Loc);
7376template <
class Emitter>
7377bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
7389 return this->emitPopPtr(Loc);
7391 for (ssize_t I = N - 1; I >= 1; --I) {
7392 if (!this->emitConstUint64(I, Loc))
7394 if (!this->emitArrayElemPtrUint64(Loc))
7396 if (!this->emitDestructionPop(ElemDesc, Loc))
7400 if (!this->emitConstUint64(0, Loc))
7402 if (!this->emitArrayElemPtrPopUint64(Loc))
7404 return this->emitDestructionPop(ElemDesc, Loc);
7409 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
7414template <
class Emitter>
7415bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E) {
7416 assert(!DiscardResult &&
"Should've been checked before");
7418 unsigned DummyID = P.getOrCreateDummy(D);
7420 if (!this->emitGetPtrGlobal(DummyID, E))
7428 return this->emitDecayPtr(
PT_Ptr, PT, E);
7434template <
class Emitter>
7435bool Compiler<Emitter>::emitFloat(
const APFloat &F,
const Expr *E) {
7436 assert(!DiscardResult &&
"Should've been checked before");
7439 return this->emitConstFloat(
Floating(F), E);
7441 APInt I = F.bitcastToAPInt();
7442 return this->emitConstFloat(
7443 Floating(
const_cast<uint64_t *
>(I.getRawData()),
7444 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
7455template <
class Emitter>
7456bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
7469 if (!this->emitGetPtrLocal(*LocalIndex, E))
7479 if (!this->visit(SubExpr))
7481 }
else if (
OptPrimType FromT = classify(SubExpr)) {
7482 unsigned TempOffset =
7483 allocateLocalPrimitive(SubExpr, *FromT,
true);
7484 if (!this->visit(SubExpr))
7486 if (!this->emitSetLocal(*FromT, TempOffset, E))
7488 if (!this->emitGetPtrLocal(TempOffset, E))
7495 if (!this->emitBitCast(E))
7497 return DiscardResult ? this->emitPopPtr(E) :
true;
7501 const llvm::fltSemantics *TargetSemantics =
nullptr;
7503 TargetSemantics = &Ctx.getFloatSemantics(ToType);
7509 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
7511 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
7512 ResultBitWidth, TargetSemantics,
7517 return this->emitPop(*ToT, 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, EHScopeStack::Cleanup *cleanup)
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)
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
const ValueDecl * getMemberPointerDecl() const
APValue & getUnionValue()
bool isMemberPointer() const
unsigned getArraySize() const
@ None
There is no such object (it's outside its lifetime).
bool isNullPointer() const
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
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 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...
CastKind getCastKind() const
llvm::iterator_range< path_iterator > path()
Path through the class hierarchy taken by casts between base and derived classes (see implementation ...
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
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.
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.
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...
const Expr * getBase() const
Represents a member of a struct/union/class.
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
llvm::APInt getValue() const
Returns an internal integer representation of the literal.
llvm::APFloat getValue() const
ForStmt - This represents a 'for (init;cond;inc)' stmt.
VarDecl * getConditionVariable() const
Retrieve the variable declared in this "for" statement, if any.
DeclStmt * getConditionVariableDeclStmt()
If this ForStmt has a condition variable, return the faux DeclStmt associated with the creation of th...
const Expr * getSubExpr() const
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
bool isFunctionTemplateSpecialization() const
Determine whether this function is a function template specialization.
FunctionTemplateDecl * getDescribedFunctionTemplate() const
Retrieves the function template that is described by this function declaration.
unsigned getBuiltinID(bool ConsiderWrapperFunctions=false) const
Returns a value indicating whether this function corresponds to a builtin function.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
const TemplateArgumentList * getTemplateSpecializationArgs() const
Retrieve the template arguments used to produce this function template specialization from the primar...
bool isUsableAsGlobalAllocationFunctionInConstantEvaluation(UnsignedOrNone *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions described in i...
bool isDefaulted() const
Whether this function is defaulted.
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Declaration of a template function.
FunctionDecl * findSpecialization(ArrayRef< TemplateArgument > Args, void *&InsertPos)
Return the specialization with the provided arguments if it exists, otherwise return the insertion po...
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
Expr * getResultExpr()
Return the result expression of this controlling expression.
IfStmt - This represents an if/then/else.
bool isNonNegatedConsteval() const
bool isNegatedConsteval() const
DeclStmt * getConditionVariableDeclStmt()
If this IfStmt has a condition variable, return the faux DeclStmt associated with the creation of tha...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "if" statement, if any.
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
const Expr * getSubExpr() const
Represents an implicitly-generated value initialization of an object of a given type.
Represents a field injected from an anonymous union/struct into the parent scope.
Describes an C or C++ initializer list.
Expr * getArrayFiller()
If this initializer list initializes an array with more elements than there are initializers in the l...
ArrayRef< Expr * > inits()
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
capture_init_iterator capture_init_begin()
Retrieve the first initialization argument for this lambda expression (which initializes the first ca...
CXXRecordDecl * getLambdaClass() const
Retrieve the class that corresponds to the lambda.
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
const Stmt * getNamedLoopOrSwitch() const
If this is a named break/continue, get the loop or switch statement that this targets.
APValue & getAsAPValue() const
Get the value of this MSGuidDecl as an APValue.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
StorageDuration getStorageDuration() const
Retrieve the storage duration for the materialized temporary.
Expr * getSubExpr() const
Retrieve the temporary-generating subexpression whose value will be materialized into a glvalue.
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.
CXXRecordDecl * getMostRecentCXXRecordDecl() const
Note: this can trigger extra deserialization when external AST sources are used.
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.
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
bool isExpressibleAsConstantInitializer() const
ObjCEncodeExpr, used for @encode in Objective-C.
QualType getEncodedType() const
SourceLocation getAtLoc() 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
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 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.
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.
std::byte * rawData()
Returns a pointer to the raw data, including metadata.
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.
bool VisitCXXParenListInitExpr(const CXXParenListInitExpr *E)
bool VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E)
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)
VarCreationState visitDecl(const VarDecl *VD, bool IsConstexprUnknown=false)
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.
void emitCleanup()
Emits scope cleanup instructions.
bool VisitFixedPointBinOp(const BinaryOperator *E)
bool VisitCastExpr(const CastExpr *E)
bool VisitObjCEncodeExpr(const ObjCEncodeExpr *E)
bool VisitFixedPointUnaryOperator(const UnaryOperator *E)
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)
VariableScope< Emitter > * VarScope
Current scope.
bool visitDeclAndReturn(const VarDecl *VD, const Expr *Init, bool ConstantContext) override
Toplevel visitDeclAndReturn().
bool VisitCXXNewExpr(const CXXNewExpr *E)
const ValueDecl * InitializingDecl
bool VisitCompoundAssignOperator(const CompoundAssignOperator *E)
bool visit(const Expr *E) override
Evaluates an expression and places the result on the stack.
bool delegate(const Expr *E)
Just pass evaluation on to E.
bool discard(const Expr *E)
Evaluates an expression for side effects and discards the result.
bool VisitCXXDefaultArgExpr(const CXXDefaultArgExpr *E)
CaseMap CaseLabels
Switch case mapping.
Record * getRecord(QualType Ty)
Returns a record from a record or pointer type.
const RecordType * getRecordTy(QualType Ty)
Returns a record type from a record or pointer type.
bool VisitCXXStdInitializerListExpr(const CXXStdInitializerListExpr *E)
bool visitInitList(ArrayRef< const Expr * > Inits, const Expr *ArrayFiller, const Expr *E)
bool VisitSizeOfPackExpr(const SizeOfPackExpr *E)
bool VisitPredefinedExpr(const PredefinedExpr *E)
bool VisitSourceLocExpr(const SourceLocExpr *E)
bool visitDeclStmt(const DeclStmt *DS, bool EvaluateConditionDecl=false)
bool 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.
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)
typename Emitter::LabelTy LabelTy
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)
VarCreationState visitVarDecl(const VarDecl *VD, const Expr *Init, bool Toplevel=false, bool IsConstexprUnknown=false)
Creates and initializes a variable from the given decl.
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)
unsigned allocateLocalPrimitive(DeclTy &&Decl, PrimType Ty, bool IsConst, bool IsVolatile=false, const ValueDecl *ExtendingDecl=nullptr, ScopeKind SC=ScopeKind::Block, bool IsConstexprUnknown=false)
Creates a local primitive value.
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)
UnsignedOrNone allocateLocal(DeclTy &&Decl, QualType Ty=QualType(), const ValueDecl *ExtendingDecl=nullptr, ScopeKind=ScopeKind::Block, bool IsConstexprUnknown=false)
Allocates a space storing a local given its type.
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 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 CXXDestructorDecl * getDestructor() const
Returns the destructor of the record, if any.
const Field * getField(const FieldDecl *FD) const
Returns a field.
bool hasTrivialDtor() const
Returns true for anonymous unions and records with no destructor or for those with a trivial destruct...
llvm::iterator_range< const_base_iter > bases() const
const Base * getVirtualBase(const RecordDecl *RD) const
Returns a virtual base descriptor.
unsigned getNumFields() const
llvm::iterator_range< const_field_iter > fields() const
const Base * getBase(const RecordDecl *FD) const
Returns a base descriptor.
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(const 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)
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)
constexpr bool isIntegralType(PrimType T)
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
@ Success
Annotation was successful.
UnaryExprOrTypeTrait
Names for the "expression or type" traits.
@ SD_Static
Static storage duration.
@ Result
The result type of a method or function.
const FunctionProtoType * T
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.
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)
bool isUnnamedBitField() const
Information about a local's storage.
State encapsulating if a the variable creation has been successful, unsuccessful, or no variable has ...
static VarCreationState NotCreated()