33 OldInitializingDecl(
Ctx->InitializingDecl) {
34 Ctx->InitializingDecl = VD;
43 this->
Ctx->InitializingDecl = OldInitializingDecl;
44 this->
Ctx->InitStack.pop_back();
58 : Ctx(Ctx), OldDiscardResult(Ctx->DiscardResult),
65 Ctx->DiscardResult = OldDiscardResult;
66 Ctx->Initializing = OldInitializing;
73 bool OldDiscardResult;
77template <
class Emitter>
81 return Ctx->emitThis(
E);
84 return Ctx->emitGetPtrFieldPop(
Offset,
E);
86 return Ctx->emitGetPtrLocal(
Offset,
E);
90 if (!Ctx->emitConstUint32(
Offset,
E))
92 return Ctx->emitArrayElemPtrPopUint32(
E);
94 return Ctx->emitRVOPtr(
E);
98 llvm_unreachable(
"Unhandled InitLink kind");
122 OldContinueLabel(
Ctx->ContinueLabel),
123 OldBreakVarScope(
Ctx->BreakVarScope),
124 OldContinueVarScope(
Ctx->ContinueVarScope) {
132 this->
Ctx->BreakLabel = OldBreakLabel;
133 this->
Ctx->ContinueLabel = OldContinueLabel;
134 this->
Ctx->ContinueVarScope = OldContinueVarScope;
135 this->
Ctx->BreakVarScope = OldBreakVarScope;
155 OldDefaultLabel(this->
Ctx->DefaultLabel),
156 OldCaseLabels(
std::move(this->
Ctx->CaseLabels)),
157 OldLabelVarScope(
Ctx->BreakVarScope) {
160 this->Ctx->
CaseLabels = std::move(CaseLabels);
165 this->
Ctx->BreakLabel = OldBreakLabel;
166 this->
Ctx->DefaultLabel = OldDefaultLabel;
167 this->
Ctx->CaseLabels = std::move(OldCaseLabels);
168 this->
Ctx->BreakVarScope = OldLabelVarScope;
194template <
class Emitter>
199 case CK_LValueToRValue: {
201 return this->discard(SubExpr);
203 std::optional<PrimType> SubExprT = classify(SubExpr->
getType());
206 std::optional<unsigned> LocalIndex = allocateLocal(SubExpr);
209 if (!this->emitGetPtrLocal(*LocalIndex, CE))
213 if (!this->visit(SubExpr))
217 return this->emitLoadPop(*SubExprT, CE);
222 return this->emitMemcpy(CE);
225 case CK_DerivedToBaseMemberPointer: {
234 if (!this->delegate(SubExpr))
237 return this->emitGetMemberPtrBasePop(DerivedOffset, CE);
240 case CK_BaseToDerivedMemberPointer: {
246 unsigned DerivedOffset = collectBaseOffset(
QualType(FromMP->getClass(), 0),
249 if (!this->delegate(SubExpr))
251 return this->emitGetMemberPtrBasePop(-DerivedOffset, CE);
254 case CK_UncheckedDerivedToBase:
255 case CK_DerivedToBase: {
256 if (!this->delegate(SubExpr))
260 if (
const auto *PT = dyn_cast<PointerType>(Ty))
261 return PT->getPointeeType()->getAsCXXRecordDecl();
262 return Ty->getAsCXXRecordDecl();
269 if (B->isVirtual()) {
270 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), CE))
272 CurType = B->getType();
274 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
275 if (!this->emitGetPtrBasePop(DerivedOffset, CE))
277 CurType = B->getType();
284 case CK_BaseToDerived: {
285 if (!this->delegate(SubExpr))
288 unsigned DerivedOffset =
291 return this->emitGetPtrDerivedPop(DerivedOffset, CE);
294 case CK_FloatingCast: {
300 return this->discard(SubExpr);
301 if (!this->visit(SubExpr))
303 const auto *TargetSemantics = &Ctx.getFloatSemantics(CE->
getType());
307 case CK_IntegralToFloating: {
309 return this->discard(SubExpr);
310 std::optional<PrimType> FromT = classify(SubExpr->
getType());
314 if (!this->visit(SubExpr))
317 const auto *TargetSemantics = &Ctx.getFloatSemantics(CE->
getType());
318 return this->emitCastIntegralFloating(*FromT, TargetSemantics,
319 getFPOptions(CE), CE);
322 case CK_FloatingToBoolean:
323 case CK_FloatingToIntegral: {
325 return this->discard(SubExpr);
327 std::optional<PrimType> ToT = classify(CE->
getType());
332 if (!this->visit(SubExpr))
336 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(CE->
getType()),
337 getFPOptions(CE), CE);
339 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(CE->
getType()),
340 getFPOptions(CE), CE);
342 return this->emitCastFloatingIntegral(*ToT, getFPOptions(CE), CE);
345 case CK_NullToPointer:
346 case CK_NullToMemberPointer: {
347 if (!this->discard(SubExpr))
354 if (!PointeeType.
isNull()) {
355 if (std::optional<PrimType>
T = classify(PointeeType))
356 Desc =
P.createDescriptor(SubExpr, *
T);
358 Desc =
P.createDescriptor(SubExpr, PointeeType.
getTypePtr(),
359 std::nullopt,
true,
false,
363 uint64_t Val = Ctx.getASTContext().getTargetNullPointerValue(CE->
getType());
364 return this->emitNull(classifyPrim(CE->
getType()), Val, Desc, CE);
367 case CK_PointerToIntegral: {
369 return this->discard(SubExpr);
371 if (!this->visit(SubExpr))
377 if (!this->emitDecayPtr(FromT,
PT_Ptr, CE))
383 return this->emitCastPointerIntegralAP(Ctx.getBitWidth(CE->
getType()),
386 return this->emitCastPointerIntegralAPS(Ctx.getBitWidth(CE->
getType()),
388 return this->emitCastPointerIntegral(
T, CE);
391 case CK_ArrayToPointerDecay: {
392 if (!this->visit(SubExpr))
394 if (!this->emitArrayDecay(CE))
397 return this->emitPopPtr(CE);
401 case CK_IntegralToPointer: {
404 if (!this->visit(SubExpr))
410 return this->emitPop(
T, CE);
415 Desc =
P.createDescriptor(SubExpr, *
T);
423 if (!this->emitGetIntPtr(
T, Desc, CE))
426 PrimType DestPtrT = classifyPrim(PtrType);
431 return this->emitDecayPtr(
PT_Ptr, DestPtrT, CE);
434 case CK_AtomicToNonAtomic:
435 case CK_ConstructorConversion:
436 case CK_FunctionToPointerDecay:
437 case CK_NonAtomicToAtomic:
439 case CK_UserDefinedConversion:
440 case CK_AddressSpaceConversion:
441 case CK_CPointerToObjCPointerCast:
442 return this->delegate(SubExpr);
447 if (!this->discard(SubExpr))
449 return this->emitInvalidCast(CastKind::Reinterpret,
true, CE);
453 return this->discard(SubExpr);
456 std::optional<PrimType> FromT = classify(SubExprTy);
459 return this->emitBuiltinBitCast(CE);
461 std::optional<PrimType> ToT = classify(CE->
getType());
469 return this->delegate(SubExpr);
471 if (!this->visit(SubExpr))
478 if (!this->visit(SubExpr))
480 return this->emitDecayPtr(*FromT, *ToT, CE);
483 case CK_LValueToRValueBitCast:
484 return this->emitBuiltinBitCast(CE);
486 case CK_IntegralToBoolean:
487 case CK_FixedPointToBoolean:
488 case CK_BooleanToSignedIntegral:
489 case CK_IntegralCast: {
491 return this->discard(SubExpr);
492 std::optional<PrimType> FromT = classify(SubExpr->
getType());
493 std::optional<PrimType> ToT = classify(CE->
getType());
498 if (!this->visit(SubExpr))
506 if (!this->emitCheckEnumValue(*FromT, ET->getDecl(), CE))
511 auto maybeNegate = [&]() ->
bool {
512 if (CE->
getCastKind() == CK_BooleanToSignedIntegral)
513 return this->emitNeg(*ToT, CE);
518 return this->emitCastAP(*FromT, Ctx.getBitWidth(CE->
getType()), CE) &&
521 return this->emitCastAPS(*FromT, Ctx.getBitWidth(CE->
getType()), CE) &&
526 if (!this->emitCast(*FromT, *ToT, CE))
529 return maybeNegate();
532 case CK_PointerToBoolean:
533 case CK_MemberPointerToBoolean: {
536 if (!this->visit(SubExpr))
538 return this->emitIsNonNull(PtrT, CE);
541 case CK_IntegralComplexToBoolean:
542 case CK_FloatingComplexToBoolean: {
544 return this->discard(SubExpr);
545 if (!this->visit(SubExpr))
547 return this->emitComplexBoolCast(SubExpr);
550 case CK_IntegralComplexToReal:
551 case CK_FloatingComplexToReal:
552 return this->emitComplexReal(SubExpr);
554 case CK_IntegralRealToComplex:
555 case CK_FloatingRealToComplex: {
559 unsigned LocalIndex = allocateTemporary(CE);
560 if (!this->emitGetPtrLocal(LocalIndex, CE))
565 if (!this->visitArrayElemInit(0, SubExpr))
569 if (!this->visitZeroInitializer(
T, SubExpr->
getType(), SubExpr))
571 return this->emitInitElem(
T, 1, SubExpr);
574 case CK_IntegralComplexCast:
575 case CK_FloatingComplexCast:
576 case CK_IntegralComplexToFloatingComplex:
577 case CK_FloatingComplexToIntegralComplex: {
581 return this->discard(SubExpr);
584 std::optional<unsigned> LocalIndex = allocateLocal(CE);
587 if (!this->emitGetPtrLocal(*LocalIndex, CE))
594 unsigned SubExprOffset = allocateLocalPrimitive(
595 SubExpr,
PT_Ptr,
true,
false);
596 if (!this->visit(SubExpr))
598 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, CE))
604 PrimType DestElemT = classifyPrim(DestElemType);
606 for (
unsigned I = 0; I != 2; ++I) {
607 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, CE))
609 if (!this->emitArrayElemPop(SourceElemT, I, CE))
613 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, CE))
617 if (!this->emitInitElem(DestElemT, I, CE))
623 case CK_VectorSplat: {
624 assert(!classify(CE->
getType()));
625 assert(classify(SubExpr->
getType()));
629 return this->discard(SubExpr);
632 std::optional<unsigned> LocalIndex = allocateLocal(CE);
635 if (!this->emitGetPtrLocal(*LocalIndex, CE))
641 unsigned ElemOffset = allocateLocalPrimitive(
642 SubExpr, ElemT,
true,
false);
645 if (!this->visit(SubExpr))
647 if (classifyPrim(SubExpr) ==
PT_Ptr && !this->emitLoadPop(ElemT, CE))
650 if (!this->emitSetLocal(ElemT, ElemOffset, CE))
653 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
654 if (!this->emitGetLocal(ElemT, ElemOffset, CE))
656 if (!this->emitInitElem(ElemT, I, CE))
663 case CK_HLSLVectorTruncation: {
665 if (std::optional<PrimType> ResultT = classify(CE)) {
666 assert(!DiscardResult);
668 if (!this->visit(SubExpr))
670 return this->emitArrayElemPop(*ResultT, 0, CE);
676 unsigned LocalIndex = allocateTemporary(CE);
677 if (!this->emitGetPtrLocal(LocalIndex, CE))
682 if (!this->visit(SubExpr))
684 return this->emitCopyArray(classifyVectorElementType(CE->
getType()), 0, 0,
688 case CK_IntegralToFixedPoint: {
689 if (!this->visit(SubExpr))
692 auto Sem = Ctx.getASTContext().getFixedPointSemantics(CE->
getType());
694 std::memcpy(&I, &Sem,
sizeof(Sem));
695 return this->emitCastIntegralFixedPoint(classifyPrim(SubExpr->
getType()), I,
698 case CK_FloatingToFixedPoint: {
699 if (!this->visit(SubExpr))
702 auto Sem = Ctx.getASTContext().getFixedPointSemantics(CE->
getType());
704 std::memcpy(&I, &Sem,
sizeof(Sem));
705 return this->emitCastFloatingFixedPoint(I, CE);
707 case CK_FixedPointToFloating: {
708 if (!this->visit(SubExpr))
710 const auto *TargetSemantics = &Ctx.getFloatSemantics(CE->
getType());
711 return this->emitCastFixedPointFloating(TargetSemantics, CE);
713 case CK_FixedPointToIntegral: {
714 if (!this->visit(SubExpr))
716 return this->emitCastFixedPointIntegral(classifyPrim(CE->
getType()), CE);
718 case CK_FixedPointCast: {
719 if (!this->visit(SubExpr))
721 auto Sem = Ctx.getASTContext().getFixedPointSemantics(CE->
getType());
723 std::memcpy(&I, &Sem,
sizeof(Sem));
724 return this->emitCastFixedPoint(I, CE);
728 return discard(SubExpr);
731 return this->emitInvalid(CE);
733 llvm_unreachable(
"Unhandled clang::CastKind enum");
736template <
class Emitter>
741 return this->emitConst(
LE->getValue(),
LE);
744template <
class Emitter>
749 return this->emitConstFloat(
E->getValue(),
E);
752template <
class Emitter>
759 unsigned LocalIndex = allocateTemporary(
E);
760 if (!this->emitGetPtrLocal(LocalIndex,
E))
764 const Expr *SubExpr =
E->getSubExpr();
767 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
769 if (!this->emitInitElem(SubExprT, 0, SubExpr))
771 return this->visitArrayElemInit(1, SubExpr);
774template <
class Emitter>
782 auto Sem = Ctx.getASTContext().getFixedPointSemantics(
E->
getType());
787template <
class Emitter>
789 return this->delegate(
E->getSubExpr());
792template <
class Emitter>
796 return this->VisitLogicalBinOp(BO);
804 if (!this->discard(LHS))
807 return this->discard(RHS);
809 return this->delegate(RHS);
813 return this->VisitComplexBinOp(BO);
815 return this->VisitVectorBinOp(BO);
819 return this->emitComplexComparison(LHS, RHS, BO);
821 return this->VisitFixedPointBinOp(BO);
824 if (!this->visit(LHS))
827 if (!this->visit(RHS))
830 if (!this->emitToMemberPtr(BO))
836 if (!this->emitCastMemberPtrPtr(BO))
838 return DiscardResult ? this->emitPopPtr(BO) :
true;
842 std::optional<PrimType>
LT = classify(LHS);
843 std::optional<PrimType> RT = classify(RHS);
844 std::optional<PrimType>
T = classify(BO->
getType());
853 Ctx.getASTContext().CompCategories.lookupInfoForType(BO->
getType());
858 std::optional<unsigned> ResultIndex = this->allocateLocal(BO);
859 if (!this->emitGetPtrLocal(*ResultIndex, BO))
863 if (!visit(LHS) || !visit(RHS))
866 return this->emitCMP3(*
LT, CmpInfo, BO);
869 if (!
LT || !RT || !
T)
875 return this->VisitPointerArithBinOp(BO);
880 if (!visit(RHS) || !visit(LHS))
882 if (!this->emitFlip(*
LT, *RT, BO))
885 if (!visit(LHS) || !visit(RHS))
891 auto MaybeCastToBool = [
this,
T, BO](
bool Result) {
895 return this->emitPop(*
T, BO);
897 return this->emitCast(
PT_Bool, *
T, BO);
901 auto Discard = [
this,
T, BO](
bool Result) {
904 return DiscardResult ? this->emitPop(*
T, BO) :
true;
909 return MaybeCastToBool(this->emitEQ(*
LT, BO));
911 return MaybeCastToBool(this->emitNE(*
LT, BO));
913 return MaybeCastToBool(this->emitLT(*
LT, BO));
915 return MaybeCastToBool(this->emitLE(*
LT, BO));
917 return MaybeCastToBool(this->emitGT(*
LT, BO));
919 return MaybeCastToBool(this->emitGE(*
LT, BO));
922 return Discard(this->emitSubf(getFPOptions(BO), BO));
923 return Discard(this->emitSub(*
T, BO));
926 return Discard(this->emitAddf(getFPOptions(BO), BO));
927 return Discard(this->emitAdd(*
T, BO));
930 return Discard(this->emitMulf(getFPOptions(BO), BO));
931 return Discard(this->emitMul(*
T, BO));
933 return Discard(this->emitRem(*
T, BO));
936 return Discard(this->emitDivf(getFPOptions(BO), BO));
937 return Discard(this->emitDiv(*
T, BO));
941 : this->emitStorePop(*
T, BO);
943 if (!this->emitStoreBitField(*
T, BO))
946 if (!this->emitStore(*
T, BO))
952 return this->emitLoadPop(*
T, BO);
955 return Discard(this->emitBitAnd(*
T, BO));
957 return Discard(this->emitBitOr(*
T, BO));
959 return Discard(this->emitShl(*
LT, *RT, BO));
961 return Discard(this->emitShr(*
LT, *RT, BO));
963 return Discard(this->emitBitXor(*
T, BO));
966 llvm_unreachable(
"Already handled earlier");
971 llvm_unreachable(
"Unhandled binary op");
976template <
class Emitter>
979 const Expr *LHS =
E->getLHS();
980 const Expr *RHS =
E->getRHS();
982 if ((Op != BO_Add && Op != BO_Sub) ||
986 std::optional<PrimType>
LT = classify(LHS);
987 std::optional<PrimType> RT = classify(RHS);
997 return this->emitDecayPtr(
T,
PT_Ptr,
E);
1006 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1010 if (!this->emitSubPtr(IntT,
E))
1012 return DiscardResult ? this->emitPop(IntT,
E) :
true;
1017 if (!visitAsPointer(RHS, *RT))
1019 if (!this->visit(LHS))
1023 if (!visitAsPointer(LHS, *
LT))
1025 if (!this->visit(RHS))
1035 if (!this->emitAddOffset(OffsetType,
E))
1038 if (classifyPrim(
E) !=
PT_Ptr)
1039 return this->emitDecayPtr(
PT_Ptr, classifyPrim(
E),
E);
1041 }
else if (Op == BO_Sub) {
1042 if (!this->emitSubOffset(OffsetType,
E))
1045 if (classifyPrim(
E) !=
PT_Ptr)
1046 return this->emitDecayPtr(
PT_Ptr, classifyPrim(
E),
E);
1053template <
class Emitter>
1055 assert(
E->isLogicalOp());
1057 const Expr *LHS =
E->getLHS();
1058 const Expr *RHS =
E->getRHS();
1059 std::optional<PrimType>
T = classify(
E->
getType());
1063 LabelTy LabelTrue = this->getLabel();
1064 LabelTy LabelEnd = this->getLabel();
1066 if (!this->visitBool(LHS))
1068 if (!this->jumpTrue(LabelTrue))
1071 if (!this->visitBool(RHS))
1073 if (!this->jump(LabelEnd))
1076 this->emitLabel(LabelTrue);
1077 this->emitConstBool(
true,
E);
1078 this->fallthrough(LabelEnd);
1079 this->emitLabel(LabelEnd);
1082 assert(Op == BO_LAnd);
1085 LabelTy LabelFalse = this->getLabel();
1086 LabelTy LabelEnd = this->getLabel();
1088 if (!this->visitBool(LHS))
1090 if (!this->jumpFalse(LabelFalse))
1093 if (!this->visitBool(RHS))
1095 if (!this->jump(LabelEnd))
1098 this->emitLabel(LabelFalse);
1099 this->emitConstBool(
false,
E);
1100 this->fallthrough(LabelEnd);
1101 this->emitLabel(LabelEnd);
1105 return this->emitPopBool(
E);
1114template <
class Emitter>
1118 unsigned LocalIndex = allocateTemporary(
E);
1119 if (!this->emitGetPtrLocal(LocalIndex,
E))
1125 const Expr *LHS =
E->getLHS();
1126 const Expr *RHS =
E->getRHS();
1129 unsigned ResultOffset = ~0u;
1131 ResultOffset = this->allocateLocalPrimitive(
E,
PT_Ptr,
true,
false);
1134 if (!this->DiscardResult) {
1135 if (!this->emitDupPtr(
E))
1137 if (!this->emitSetLocal(
PT_Ptr, ResultOffset,
E))
1142 LHSType = AT->getValueType();
1145 RHSType = AT->getValueType();
1154 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1159 if (!this->visit(LHS))
1161 if (!this->visit(RHS))
1163 return this->emitMulc(ElemT,
E);
1166 if (Op == BO_Div && RHSIsComplex) {
1168 PrimType ElemT = classifyPrim(ElemQT);
1173 if (!LHSIsComplex) {
1175 LHSOffset = allocateTemporary(RHS);
1177 if (!this->emitGetPtrLocal(LHSOffset,
E))
1180 if (!this->visit(LHS))
1183 if (!this->emitInitElem(ElemT, 0,
E))
1186 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
1188 if (!this->emitInitElem(ElemT, 1,
E))
1191 if (!this->visit(LHS))
1195 if (!this->visit(RHS))
1197 return this->emitDivc(ElemT,
E);
1202 LHSOffset = this->allocateLocalPrimitive(LHS,
PT_Ptr,
true,
false);
1203 if (!this->visit(LHS))
1205 if (!this->emitSetLocal(
PT_Ptr, LHSOffset,
E))
1208 PrimType LHST = classifyPrim(LHSType);
1209 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true,
false);
1210 if (!this->visit(LHS))
1212 if (!this->emitSetLocal(LHST, LHSOffset,
E))
1219 RHSOffset = this->allocateLocalPrimitive(RHS,
PT_Ptr,
true,
false);
1220 if (!this->visit(RHS))
1222 if (!this->emitSetLocal(
PT_Ptr, RHSOffset,
E))
1225 PrimType RHST = classifyPrim(RHSType);
1226 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true,
false);
1227 if (!this->visit(RHS))
1229 if (!this->emitSetLocal(RHST, RHSOffset,
E))
1236 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1237 unsigned ElemIndex,
unsigned Offset,
1238 const Expr *
E) ->
bool {
1240 if (!this->emitGetLocal(
PT_Ptr, Offset,
E))
1242 return this->emitArrayElemPop(classifyComplexElementType(
E->
getType()),
1245 if (ElemIndex == 0 || !LoadZero)
1246 return this->emitGetLocal(classifyPrim(
E->
getType()), Offset,
E);
1247 return this->visitZeroInitializer(classifyPrim(
E->
getType()),
E->
getType(),
1252 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1254 if (!this->DiscardResult) {
1255 if (!this->emitGetLocal(
PT_Ptr, ResultOffset,
E))
1262 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1265 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1268 if (!this->emitAddf(getFPOptions(
E),
E))
1271 if (!this->emitAdd(ResultElemT,
E))
1276 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1279 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1282 if (!this->emitSubf(getFPOptions(
E),
E))
1285 if (!this->emitSub(ResultElemT,
E))
1290 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1293 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1297 if (!this->emitMulf(getFPOptions(
E),
E))
1300 if (!this->emitMul(ResultElemT,
E))
1305 assert(!RHSIsComplex);
1306 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1309 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1313 if (!this->emitDivf(getFPOptions(
E),
E))
1316 if (!this->emitDiv(ResultElemT,
E))
1325 if (!this->DiscardResult) {
1327 if (!this->emitInitElemPop(ResultElemT, ElemIndex,
E))
1330 if (!this->emitPop(ResultElemT,
E))
1337template <
class Emitter>
1339 assert(!
E->isCommaOp() &&
1340 "Comma op should be handled in VisitBinaryOperator");
1347 unsigned LocalIndex = allocateTemporary(
E);
1348 if (!this->emitGetPtrLocal(LocalIndex,
E))
1352 const Expr *LHS =
E->getLHS();
1353 const Expr *RHS =
E->getRHS();
1355 auto Op =
E->isCompoundAssignmentOp()
1364 unsigned LHSOffset = this->allocateLocalPrimitive(LHS,
PT_Ptr,
true,
false);
1365 if (!this->visit(LHS))
1367 if (!this->emitSetLocal(
PT_Ptr, LHSOffset,
E))
1371 unsigned RHSOffset = this->allocateLocalPrimitive(RHS,
PT_Ptr,
true,
false);
1372 if (!this->visit(RHS))
1374 if (!this->emitSetLocal(
PT_Ptr, RHSOffset,
E))
1377 if (
E->isCompoundAssignmentOp() && !this->emitGetLocal(
PT_Ptr, LHSOffset,
E))
1382 bool NeedIntPromot = ElemT ==
PT_Bool && (
E->isBitwiseOp() ||
E->isShiftOp());
1384 Ctx.getASTContext().getPromotedIntegerType(Ctx.getASTContext().BoolTy);
1385 PrimType PromotT = classifyPrim(PromotTy);
1386 PrimType OpT = NeedIntPromot ? PromotT : ElemT;
1388 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1389 if (!this->emitGetLocal(
PT_Ptr, Offset,
E))
1391 if (!this->emitArrayElemPop(ElemT, Index,
E))
1393 if (
E->isLogicalOp()) {
1394 if (!this->emitPrimCast(ElemT,
PT_Bool, Ctx.getASTContext().BoolTy,
E))
1396 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(),
E))
1398 }
else if (NeedIntPromot) {
1399 if (!this->emitPrimCast(ElemT, PromotT, PromotTy,
E))
1405#define EMIT_ARITH_OP(OP) \
1407 if (ElemT == PT_Float) { \
1408 if (!this->emit##OP##f(getFPOptions(E), E)) \
1411 if (!this->emit##OP(ElemT, E)) \
1417 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1418 if (!getElem(LHSOffset, ElemT, I))
1420 if (!getElem(RHSOffset, RHSElemT, I))
1432 if (!this->emitRem(ElemT,
E))
1436 if (!this->emitBitAnd(OpT,
E))
1440 if (!this->emitBitOr(OpT,
E))
1444 if (!this->emitBitXor(OpT,
E))
1448 if (!this->emitShl(OpT, RHSElemT,
E))
1452 if (!this->emitShr(OpT, RHSElemT,
E))
1456 if (!this->emitEQ(ElemT,
E))
1460 if (!this->emitNE(ElemT,
E))
1464 if (!this->emitLE(ElemT,
E))
1468 if (!this->emitLT(ElemT,
E))
1472 if (!this->emitGE(ElemT,
E))
1476 if (!this->emitGT(ElemT,
E))
1481 if (!this->emitBitAnd(ResultElemT,
E))
1486 if (!this->emitBitOr(ResultElemT,
E))
1490 return this->emitInvalid(
E);
1498 if (
E->isComparisonOp()) {
1499 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(),
E))
1501 if (!this->emitNeg(ResultElemT,
E))
1507 if (NeedIntPromot &&
1508 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(),
E))
1512 if (!this->emitInitElem(ResultElemT, I,
E))
1516 if (DiscardResult &&
E->isCompoundAssignmentOp() && !this->emitPopPtr(
E))
1521template <
class Emitter>
1523 const Expr *LHS =
E->getLHS();
1524 const Expr *RHS =
E->getRHS();
1529 auto LHSSema = Ctx.getASTContext().getFixedPointSemantics(LHS->
getType());
1530 auto RHSSema = Ctx.getASTContext().getFixedPointSemantics(RHS->
getType());
1532 if (!this->visit(LHS))
1536 std::memcpy(&I, &LHSSema,
sizeof(llvm::FixedPointSemantics));
1537 if (!this->emitCastIntegralFixedPoint(classifyPrim(LHS->
getType()), I,
E))
1541 if (!this->visit(RHS))
1545 std::memcpy(&I, &RHSSema,
sizeof(llvm::FixedPointSemantics));
1546 if (!this->emitCastIntegralFixedPoint(classifyPrim(RHS->
getType()), I,
E))
1551 auto ConvertResult = [&](
bool R) ->
bool {
1554 auto ResultSema = Ctx.getASTContext().getFixedPointSemantics(
E->
getType());
1555 auto CommonSema = LHSSema.getCommonSemantics(RHSSema);
1556 if (ResultSema != CommonSema) {
1558 std::memcpy(&I, &ResultSema,
sizeof(ResultSema));
1559 return this->emitCastFixedPoint(I,
E);
1564 auto MaybeCastToBool = [&](
bool Result) {
1569 return this->emitPop(
T,
E);
1575 switch (
E->getOpcode()) {
1577 return MaybeCastToBool(this->emitEQFixedPoint(
E));
1579 return MaybeCastToBool(this->emitNEFixedPoint(
E));
1581 return MaybeCastToBool(this->emitLTFixedPoint(
E));
1583 return MaybeCastToBool(this->emitLEFixedPoint(
E));
1585 return MaybeCastToBool(this->emitGTFixedPoint(
E));
1587 return MaybeCastToBool(this->emitGEFixedPoint(
E));
1589 return ConvertResult(this->emitAddFixedPoint(
E));
1591 return ConvertResult(this->emitSubFixedPoint(
E));
1593 return ConvertResult(this->emitMulFixedPoint(
E));
1595 return ConvertResult(this->emitDivFixedPoint(
E));
1597 return ConvertResult(this->emitShiftFixedPoint(
true,
E));
1599 return ConvertResult(this->emitShiftFixedPoint(
false,
E));
1602 return this->emitInvalid(
E);
1605 llvm_unreachable(
"unhandled binop opcode");
1608template <
class Emitter>
1610 const Expr *SubExpr =
E->getSubExpr();
1613 switch (
E->getOpcode()) {
1615 return this->delegate(SubExpr);
1617 if (!this->visit(SubExpr))
1619 return this->emitNegFixedPoint(
E);
1624 llvm_unreachable(
"Unhandled unary opcode");
1627template <
class Emitter>
1632 if (std::optional<PrimType>
T = classify(QT))
1633 return this->visitZeroInitializer(*
T, QT,
E);
1641 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1642 CXXRD && CXXRD->getNumVBases() > 0) {
1647 const Record *R = getRecord(QT);
1652 return this->visitZeroRecordInitializer(R,
E);
1659 return this->visitZeroArrayInitializer(QT,
E);
1663 QualType ElemQT = ComplexTy->getElementType();
1664 PrimType ElemT = classifyPrim(ElemQT);
1665 for (
unsigned I = 0; I < 2; ++I) {
1666 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
1668 if (!this->emitInitElem(ElemT, I,
E))
1675 unsigned NumVecElements = VecT->getNumElements();
1676 QualType ElemQT = VecT->getElementType();
1677 PrimType ElemT = classifyPrim(ElemQT);
1679 for (
unsigned I = 0; I < NumVecElements; ++I) {
1680 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
1682 if (!this->emitInitElem(ElemT, I,
E))
1691template <
class Emitter>
1693 const Expr *LHS =
E->getLHS();
1694 const Expr *RHS =
E->getRHS();
1695 const Expr *Index =
E->getIdx();
1698 return this->discard(LHS) && this->discard(RHS);
1703 for (
const Expr *SubExpr : {LHS, RHS}) {
1704 if (!this->visit(SubExpr))
1711 PrimType IndexT = classifyPrim(Index->getType());
1714 if (!this->emitFlip(
PT_Ptr, IndexT,
E))
1718 return this->emitArrayElemPtrPop(IndexT,
E);
1721template <
class Emitter>
1723 const Expr *ArrayFiller,
const Expr *
E) {
1728 QT = AT->getValueType();
1731 if (Inits.size() == 0)
1733 return this->emitInvalid(
E);
1737 if (DiscardResult) {
1739 if (!this->discard(
Init))
1746 if (std::optional<PrimType>
T = classify(QT)) {
1747 assert(!DiscardResult);
1748 if (Inits.size() == 0)
1749 return this->visitZeroInitializer(*
T, QT,
E);
1750 assert(Inits.size() == 1);
1751 return this->delegate(Inits[0]);
1755 const Record *R = getRecord(QT);
1757 if (Inits.size() == 1 &&
E->
getType() == Inits[0]->getType())
1758 return this->delegate(Inits[0]);
1760 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
1764 if (!this->visit(
Init))
1767 if (FieldToInit->isBitField())
1768 return this->emitInitBitField(
T, FieldToInit,
E);
1769 return this->emitInitField(
T, FieldToInit->Offset,
E);
1772 auto initCompositeField = [=](
const Record::Field *FieldToInit,
1779 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
1781 if (!this->visitInitializer(
Init))
1783 return this->emitPopPtr(
E);
1787 if (Inits.size() == 0) {
1788 if (!this->visitZeroRecordInitializer(R,
E))
1793 if (
const auto *ILE = dyn_cast<InitListExpr>(
E))
1794 FToInit = ILE->getInitializedFieldInUnion();
1796 FToInit = cast<CXXParenListInitExpr>(
E)->getInitializedFieldInUnion();
1798 const Record::Field *FieldToInit = R->
getField(FToInit);
1799 if (std::optional<PrimType>
T = classify(
Init)) {
1800 if (!initPrimitiveField(FieldToInit,
Init, *
T))
1803 if (!initCompositeField(FieldToInit,
Init))
1807 return this->emitFinishInit(
E);
1811 unsigned InitIndex = 0;
1814 while (InitIndex < R->getNumFields() &&
1818 if (std::optional<PrimType>
T = classify(
Init)) {
1819 const Record::Field *FieldToInit = R->
getField(InitIndex);
1820 if (!initPrimitiveField(FieldToInit,
Init, *
T))
1825 if (
const Record::Base *B = R->
getBase(
Init->getType())) {
1826 if (!this->emitGetPtrBase(B->Offset,
Init))
1829 if (!this->visitInitializer(
Init))
1832 if (!this->emitFinishInitPop(
E))
1837 const Record::Field *FieldToInit = R->
getField(InitIndex);
1838 if (!initCompositeField(FieldToInit,
Init))
1844 return this->emitFinishInit(
E);
1848 if (Inits.size() == 1 && QT == Inits[0]->getType())
1849 return this->delegate(Inits[0]);
1851 unsigned ElementIndex = 0;
1853 if (
const auto *EmbedS =
1854 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
1857 auto Eval = [&](
const Expr *
Init,
unsigned ElemIndex) {
1859 if (!this->visit(
Init))
1861 if (InitT != TargetT) {
1862 if (!this->emitCast(InitT, TargetT,
E))
1865 return this->emitInitElem(TargetT, ElemIndex,
Init);
1867 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
1870 if (!this->visitArrayElemInit(ElementIndex,
Init))
1880 Ctx.getASTContext().getAsConstantArrayType(QT);
1883 for (; ElementIndex != NumElems; ++ElementIndex) {
1884 if (!this->visitArrayElemInit(ElementIndex, ArrayFiller))
1889 return this->emitFinishInit(
E);
1893 unsigned NumInits = Inits.size();
1896 return this->delegate(Inits[0]);
1898 QualType ElemQT = ComplexTy->getElementType();
1899 PrimType ElemT = classifyPrim(ElemQT);
1900 if (NumInits == 0) {
1902 for (
unsigned I = 0; I < 2; ++I) {
1903 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
1905 if (!this->emitInitElem(ElemT, I,
E))
1908 }
else if (NumInits == 2) {
1909 unsigned InitIndex = 0;
1911 if (!this->visit(
Init))
1914 if (!this->emitInitElem(ElemT, InitIndex,
E))
1923 unsigned NumVecElements = VecT->getNumElements();
1924 assert(NumVecElements >= Inits.size());
1926 QualType ElemQT = VecT->getElementType();
1927 PrimType ElemT = classifyPrim(ElemQT);
1930 unsigned InitIndex = 0;
1932 if (!this->visit(
Init))
1937 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
1938 if (!this->emitCopyArray(ElemT, 0, InitIndex,
1939 InitVecT->getNumElements(),
E))
1941 InitIndex += InitVecT->getNumElements();
1943 if (!this->emitInitElem(ElemT, InitIndex,
E))
1949 assert(InitIndex <= NumVecElements);
1952 for (; InitIndex != NumVecElements; ++InitIndex) {
1953 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
1955 if (!this->emitInitElem(ElemT, InitIndex,
E))
1966template <
class Emitter>
1969 if (std::optional<PrimType>
T = classify(
Init->getType())) {
1971 if (!this->visit(
Init))
1973 return this->emitInitElem(*
T, ElemIndex,
Init);
1979 if (!this->emitConstUint32(ElemIndex,
Init))
1981 if (!this->emitArrayElemPtrUint32(
Init))
1983 if (!this->visitInitializer(
Init))
1985 return this->emitFinishInitPop(
Init);
1988template <
class Emitter>
1990 return this->visitInitList(
E->inits(),
E->getArrayFiller(),
E);
1993template <
class Emitter>
1996 return this->visitInitList(
E->getInitExprs(),
E->getArrayFiller(),
E);
1999template <
class Emitter>
2002 return this->delegate(
E->getReplacement());
2005template <
class Emitter>
2007 std::optional<PrimType>
T = classify(
E->
getType());
2008 if (
T &&
E->hasAPValueResult()) {
2015 if (this->visitAPValue(
E->getAPValueResult(), *
T,
E))
2018 return this->delegate(
E->getSubExpr());
2021template <
class Emitter>
2023 auto It =
E->begin();
2024 return this->visit(*It);
2029 bool AlignOfReturnsPreferred =
2030 ASTCtx.
getLangOpts().getClangABICompat() <= LangOptions::ClangABI::Ver7;
2038 if (
T.getQualifiers().hasUnaligned())
2044 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2050template <
class Emitter>
2054 const ASTContext &ASTCtx = Ctx.getASTContext();
2056 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2057 QualType ArgType =
E->getTypeOfArgument();
2071 if (Kind == UETT_SizeOf)
2080 return this->emitConst(Size.getQuantity(),
E);
2083 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2086 if (
E->isArgumentType()) {
2087 QualType ArgType =
E->getTypeOfArgument();
2100 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2103 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2113 return this->emitConst(Size.getQuantity(),
E);
2116 if (Kind == UETT_VectorElements) {
2117 if (
const auto *VT =
E->getTypeOfArgument()->getAs<
VectorType>())
2118 return this->emitConst(VT->getNumElements(),
E);
2119 assert(
E->getTypeOfArgument()->isSizelessVectorType());
2120 return this->emitSizelessVectorElementSize(
E);
2123 if (Kind == UETT_VecStep) {
2124 if (
const auto *VT =
E->getTypeOfArgument()->getAs<
VectorType>()) {
2125 unsigned N = VT->getNumElements();
2132 return this->emitConst(N,
E);
2134 return this->emitConst(1,
E);
2137 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2138 assert(
E->isArgumentType());
2147template <
class Emitter>
2154 return this->discard(
Base);
2158 const auto maybeLoadValue = [&]() ->
bool {
2161 if (std::optional<PrimType>
T = classify(
E))
2162 return this->emitLoadPop(*
T,
E);
2166 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2170 if (
auto GlobalIndex =
P.getGlobal(VD))
2171 return this->emitGetPtrGlobal(*GlobalIndex,
E) && maybeLoadValue();
2175 if (!isa<FieldDecl>(
Member)) {
2176 if (!this->discard(
Base) && !this->emitSideEffect(
E))
2179 return this->visitDeclRef(
Member,
E);
2183 if (!this->delegate(
Base))
2186 if (!this->visit(
Base))
2191 const auto *FD = cast<FieldDecl>(
Member);
2193 const Record *R = getRecord(RD);
2196 const Record::Field *F = R->
getField(FD);
2198 if (F->Decl->getType()->isReferenceType())
2199 return this->emitGetFieldPop(
PT_Ptr, F->Offset,
E) && maybeLoadValue();
2200 return this->emitGetPtrFieldPop(F->Offset,
E) && maybeLoadValue();
2203template <
class Emitter>
2209 return this->emitConst(*ArrayIndex,
E);
2212template <
class Emitter>
2215 assert(!DiscardResult);
2219 if (!this->discard(
E->getCommonExpr()))
2224 const Expr *SubExpr =
E->getSubExpr();
2225 size_t Size =
E->getArraySize().getZExtValue();
2230 for (
size_t I = 0; I != Size; ++I) {
2234 if (!this->visitArrayElemInit(I, SubExpr))
2242template <
class Emitter>
2244 const Expr *SourceExpr =
E->getSourceExpr();
2249 return this->visitInitializer(SourceExpr);
2252 if (
auto It = OpaqueExprs.find(
E); It != OpaqueExprs.end())
2253 return this->emitGetLocal(SubExprT, It->second,
E);
2255 if (!this->visit(SourceExpr))
2261 unsigned LocalIndex = allocateLocalPrimitive(
E, SubExprT,
true);
2262 if (!this->emitSetLocal(SubExprT, LocalIndex,
E))
2267 if (!DiscardResult) {
2268 if (!this->emitGetLocal(SubExprT, LocalIndex,
E))
2273 OpaqueExprs.insert({
E, LocalIndex});
2278template <
class Emitter>
2282 const Expr *TrueExpr =
E->getTrueExpr();
2283 const Expr *FalseExpr =
E->getFalseExpr();
2285 LabelTy LabelEnd = this->getLabel();
2286 LabelTy LabelFalse = this->getLabel();
2291 if (!this->jumpFalse(LabelFalse))
2296 if (!this->delegate(TrueExpr))
2298 if (!S.destroyLocals())
2302 if (!this->jump(LabelEnd))
2305 this->emitLabel(LabelFalse);
2309 if (!this->delegate(FalseExpr))
2311 if (!S.destroyLocals())
2315 this->fallthrough(LabelEnd);
2316 this->emitLabel(LabelEnd);
2321template <
class Emitter>
2327 unsigned StringIndex =
P.createGlobalString(
E);
2328 return this->emitGetPtrGlobal(StringIndex,
E);
2333 Ctx.getASTContext().getAsConstantArrayType(
E->
getType());
2334 assert(CAT &&
"a string literal that's not a constant array?");
2339 unsigned N = std::min(ArraySize,
E->getLength());
2340 size_t CharWidth =
E->getCharByteWidth();
2342 for (
unsigned I = 0; I != N; ++I) {
2343 uint32_t CodeUnit =
E->getCodeUnit(I);
2345 if (CharWidth == 1) {
2346 this->emitConstSint8(CodeUnit,
E);
2347 this->emitInitElemSint8(I,
E);
2348 }
else if (CharWidth == 2) {
2349 this->emitConstUint16(CodeUnit,
E);
2350 this->emitInitElemUint16(I,
E);
2351 }
else if (CharWidth == 4) {
2352 this->emitConstUint32(CodeUnit,
E);
2353 this->emitInitElemUint32(I,
E);
2355 llvm_unreachable(
"unsupported character width");
2360 for (
unsigned I = N; I != ArraySize; ++I) {
2361 if (CharWidth == 1) {
2362 this->emitConstSint8(0,
E);
2363 this->emitInitElemSint8(I,
E);
2364 }
else if (CharWidth == 2) {
2365 this->emitConstUint16(0,
E);
2366 this->emitInitElemUint16(I,
E);
2367 }
else if (CharWidth == 4) {
2368 this->emitConstUint32(0,
E);
2369 this->emitInitElemUint32(I,
E);
2371 llvm_unreachable(
"unsupported character width");
2378template <
class Emitter>
2382 return this->emitDummyPtr(
E,
E);
2385template <
class Emitter>
2387 auto &A = Ctx.getASTContext();
2389 A.getObjCEncodingForType(
E->getEncodedType(), Str);
2393 return this->delegate(SL);
2396template <
class Emitter>
2404 auto &A = Ctx.getASTContext();
2405 std::string ResultStr =
E->ComputeName(A);
2408 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2409 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2410 ArraySizeModifier::Normal, 0);
2414 false, ArrayTy,
E->getLocation());
2416 unsigned StringIndex =
P.createGlobalString(SL);
2417 return this->emitGetPtrGlobal(StringIndex,
E);
2420template <
class Emitter>
2424 return this->emitConst(
E->getValue(),
E);
2427template <
class Emitter>
2431 const Expr *LHS =
E->getLHS();
2432 const Expr *RHS =
E->getRHS();
2434 QualType LHSComputationType =
E->getComputationLHSType();
2435 QualType ResultType =
E->getComputationResultType();
2436 std::optional<PrimType>
LT = classify(LHSComputationType);
2437 std::optional<PrimType> RT = classify(ResultType);
2444 PrimType LHST = classifyPrim(LHSType);
2452 unsigned TempOffset = this->allocateLocalPrimitive(
E, *RT,
true);
2453 if (!this->emitSetLocal(*RT, TempOffset,
E))
2459 if (!this->emitLoad(LHST,
E))
2463 if (!this->emitPrimCast(LHST, classifyPrim(LHSComputationType),
2464 LHSComputationType,
E))
2468 if (!this->emitGetLocal(*RT, TempOffset,
E))
2471 switch (
E->getOpcode()) {
2473 if (!this->emitAddf(getFPOptions(
E),
E))
2477 if (!this->emitSubf(getFPOptions(
E),
E))
2481 if (!this->emitMulf(getFPOptions(
E),
E))
2485 if (!this->emitDivf(getFPOptions(
E),
E))
2492 if (!this->emitPrimCast(classifyPrim(ResultType), LHST, LHS->
getType(),
E))
2496 return this->emitStorePop(LHST,
E);
2497 return this->emitStore(LHST,
E);
2500template <
class Emitter>
2504 const Expr *LHS =
E->getLHS();
2505 const Expr *RHS =
E->getRHS();
2506 std::optional<PrimType>
LT = classify(LHS->
getType());
2507 std::optional<PrimType> RT = classify(RHS->
getType());
2509 if (Op != BO_AddAssign && Op != BO_SubAssign)
2518 if (!this->emitLoad(*
LT, LHS))
2524 if (Op == BO_AddAssign) {
2525 if (!this->emitAddOffset(*RT,
E))
2528 if (!this->emitSubOffset(*RT,
E))
2533 return this->emitStorePopPtr(
E);
2534 return this->emitStorePtr(
E);
2537template <
class Emitter>
2541 return VisitVectorBinOp(
E);
2543 const Expr *LHS =
E->getLHS();
2544 const Expr *RHS =
E->getRHS();
2545 std::optional<PrimType> LHSComputationT =
2546 classify(
E->getComputationLHSType());
2547 std::optional<PrimType>
LT = classify(LHS->
getType());
2548 std::optional<PrimType> RT = classify(RHS->
getType());
2549 std::optional<PrimType> ResultT = classify(
E->
getType());
2551 if (!Ctx.getLangOpts().CPlusPlus14)
2552 return this->visit(RHS) && this->visit(LHS) && this->emitError(
E);
2554 if (!
LT || !RT || !ResultT || !LHSComputationT)
2561 return VisitFloatCompoundAssignOperator(
E);
2564 return VisitPointerCompoundAssignOperator(
E);
2577 unsigned TempOffset = this->allocateLocalPrimitive(
E, *RT,
true);
2579 if (!this->emitSetLocal(*RT, TempOffset,
E))
2586 if (!this->emitLoad(*
LT,
E))
2588 if (
LT != LHSComputationT) {
2589 if (!this->emitCast(*
LT, *LHSComputationT,
E))
2594 if (!this->emitGetLocal(*RT, TempOffset,
E))
2598 switch (
E->getOpcode()) {
2600 if (!this->emitAdd(*LHSComputationT,
E))
2604 if (!this->emitSub(*LHSComputationT,
E))
2608 if (!this->emitMul(*LHSComputationT,
E))
2612 if (!this->emitDiv(*LHSComputationT,
E))
2616 if (!this->emitRem(*LHSComputationT,
E))
2620 if (!this->emitShl(*LHSComputationT, *RT,
E))
2624 if (!this->emitShr(*LHSComputationT, *RT,
E))
2628 if (!this->emitBitAnd(*LHSComputationT,
E))
2632 if (!this->emitBitXor(*LHSComputationT,
E))
2636 if (!this->emitBitOr(*LHSComputationT,
E))
2640 llvm_unreachable(
"Unimplemented compound assign operator");
2644 if (ResultT != LHSComputationT) {
2645 if (!this->emitCast(*LHSComputationT, *ResultT,
E))
2650 if (DiscardResult) {
2652 return this->emitStoreBitFieldPop(*ResultT,
E);
2653 return this->emitStorePop(*ResultT,
E);
2656 return this->emitStoreBitField(*ResultT,
E);
2657 return this->emitStore(*ResultT,
E);
2660template <
class Emitter>
2663 const Expr *SubExpr =
E->getSubExpr();
2668template <
class Emitter>
2671 const Expr *SubExpr =
E->getSubExpr();
2675 return this->delegate(SubExpr);
2680 return this->discard(SubExpr);
2684 std::optional<PrimType> SubExprT = classify(SubExpr);
2687 std::optional<unsigned> GlobalIndex =
P.createGlobal(
E);
2692 E->getLifetimeExtendedTemporaryDecl();
2697 if (!this->visit(SubExpr))
2700 if (!this->emitInitGlobalTemp(*SubExprT, *GlobalIndex, TempDecl,
E))
2703 if (!this->emitInitGlobal(*SubExprT, *GlobalIndex,
E))
2706 return this->emitGetPtrGlobal(*GlobalIndex,
E);
2709 if (!this->checkLiteralType(SubExpr))
2712 if (!this->emitGetPtrGlobal(*GlobalIndex,
E))
2714 if (!this->visitInitializer(SubExpr))
2717 return this->emitInitGlobalTempComp(TempDecl,
E);
2723 unsigned LocalIndex = allocateLocalPrimitive(
E, *SubExprT,
true,
2725 if (!this->visit(SubExpr))
2727 if (!this->emitSetLocal(*SubExprT, LocalIndex,
E))
2729 return this->emitGetPtrLocal(LocalIndex,
E);
2732 if (!this->checkLiteralType(SubExpr))
2736 if (std::optional<unsigned> LocalIndex =
2737 allocateLocal(
E, Inner->getType(),
E->getExtendingDecl())) {
2739 if (!this->emitGetPtrLocal(*LocalIndex,
E))
2741 return this->visitInitializer(SubExpr) && this->emitFinishInit(
E);
2747template <
class Emitter>
2750 return this->delegate(
E->getSubExpr());
2753template <
class Emitter>
2757 return this->discard(
Init);
2761 return this->visitInitializer(
Init) && this->emitFinishInit(
E);
2764 std::optional<PrimType>
T = classify(
E->
getType());
2765 if (
E->isFileScope()) {
2768 return this->delegate(
Init);
2770 if (std::optional<unsigned> GlobalIndex =
P.createGlobal(
E)) {
2771 if (!this->emitGetPtrGlobal(*GlobalIndex,
E))
2775 if (!this->visit(
Init))
2777 return this->emitInitGlobal(*
T, *GlobalIndex,
E);
2780 return this->visitInitializer(
Init) && this->emitFinishInit(
E);
2789 return this->delegate(
Init);
2791 unsigned LocalIndex;
2794 LocalIndex = this->allocateLocalPrimitive(
Init, *
T,
false,
false);
2795 else if (std::optional<unsigned> MaybeIndex = this->allocateLocal(
Init))
2796 LocalIndex = *MaybeIndex;
2800 if (!this->emitGetPtrLocal(LocalIndex,
E))
2804 if (!this->visit(
Init)) {
2807 return this->emitInit(*
T,
E);
2809 if (!this->visitInitializer(
Init) || !this->emitFinishInit(
E))
2818template <
class Emitter>
2823 return this->emitConstBool(
E->getValue(),
E);
2824 return this->emitConst(
E->getValue(),
E);
2827template <
class Emitter>
2831 return this->emitConst(
E->getValue(),
E);
2834template <
class Emitter>
2840 const Record *R =
P.getOrCreateRecord(
E->getLambdaClass());
2842 auto *CaptureInitIt =
E->capture_init_begin();
2845 for (
const Record::Field &F : R->
fields()) {
2852 if (std::optional<PrimType>
T = classify(
Init)) {
2853 if (!this->visit(
Init))
2856 if (!this->emitInitField(*
T, F.Offset,
E))
2859 if (!this->emitGetPtrField(F.Offset,
E))
2862 if (!this->visitInitializer(
Init))
2865 if (!this->emitPopPtr(
E))
2873template <
class Emitter>
2879 unsigned StringIndex =
P.createGlobalString(
E->getFunctionName(),
E);
2880 return this->emitGetPtrGlobal(StringIndex,
E);
2883 return this->delegate(
E->getFunctionName());
2886template <
class Emitter>
2888 if (
E->getSubExpr() && !this->discard(
E->getSubExpr()))
2891 return this->emitInvalid(
E);
2894template <
class Emitter>
2897 const Expr *SubExpr =
E->getSubExpr();
2899 std::optional<PrimType> FromT = classify(SubExpr);
2900 std::optional<PrimType> ToT = classify(
E);
2903 return this->emitInvalidCast(CastKind::Reinterpret,
true,
E);
2907 std::optional<PrimType> PointeeFromT;
2911 PointeeFromT = classify(SubExpr->
getType());
2913 std::optional<PrimType> PointeeToT;
2917 PointeeToT = classify(
E->
getType());
2920 if (PointeeToT && PointeeFromT) {
2925 if (!this->emitInvalidCast(CastKind::Reinterpret, Fatal,
E))
2928 if (
E->getCastKind() == CK_LValueBitCast)
2929 return this->delegate(SubExpr);
2930 return this->VisitCastExpr(
E);
2934 bool Fatal = (ToT != FromT);
2935 if (!this->emitInvalidCast(CastKind::Reinterpret, Fatal,
E))
2938 return this->VisitCastExpr(
E);
2941template <
class Emitter>
2947 return this->emitConstBool(
E->getValue(),
E);
2950template <
class Emitter>
2953 assert(!classify(
T));
2963 return this->visitInitializer(
E->getArg(0));
2967 if (DiscardResult) {
2971 std::optional<unsigned> LocalIndex = allocateLocal(
E);
2976 if (!this->emitGetPtrLocal(*LocalIndex,
E))
2981 if (
E->requiresZeroInitialization()) {
2984 if (!this->visitZeroRecordInitializer(R,
E))
2997 assert(
Func->hasThisPointer());
2998 assert(!
Func->hasRVO());
3002 if (!this->emitDupPtr(
E))
3006 for (
const auto *Arg :
E->arguments()) {
3007 if (!this->visit(Arg))
3011 if (
Func->isVariadic()) {
3012 uint32_t VarArgSize = 0;
3013 unsigned NumParams =
Func->getNumWrittenParams();
3014 for (
unsigned I = NumParams, N =
E->getNumArgs(); I != N; ++I) {
3018 if (!this->emitCallVar(
Func, VarArgSize,
E))
3021 if (!this->emitCall(
Func, 0,
E)) {
3026 (void)this->emitPopPtr(
E);
3032 return this->emitPopPtr(
E);
3033 return this->emitFinishInit(
E);
3038 Ctx.getASTContext().getAsConstantArrayType(
E->
getType());
3049 for (
size_t I = 0; I != NumElems; ++I) {
3050 if (!this->emitConstUint64(I,
E))
3052 if (!this->emitArrayElemPtrUint64(
E))
3056 for (
const auto *Arg :
E->arguments()) {
3057 if (!this->visit(Arg))
3061 if (!this->emitCall(
Func, 0,
E))
3070template <
class Emitter>
3076 E->EvaluateInContext(Ctx.getASTContext(), SourceLocDefaultExpr);
3080 assert(Val.
isInt());
3082 return this->emitConst(I,
E);
3089 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3090 return this->visit(LValueExpr);
3099 auto *UGCD = cast<UnnamedGlobalConstantDecl>(BaseDecl);
3101 std::optional<unsigned> GlobalIndex =
P.getOrCreateGlobal(UGCD);
3105 if (!this->emitGetPtrGlobal(*GlobalIndex,
E))
3109 const APValue &
V = UGCD->getValue();
3110 for (
unsigned I = 0, N = R->
getNumFields(); I != N; ++I) {
3111 const Record::Field *F = R->
getField(I);
3112 const APValue &FieldValue =
V.getStructField(I);
3114 PrimType FieldT = classifyPrim(F->Decl->getType());
3116 if (!this->visitAPValue(FieldValue, FieldT,
E))
3118 if (!this->emitInitField(FieldT, F->Offset,
E))
3126template <
class Emitter>
3128 unsigned N =
E->getNumComponents();
3132 for (
unsigned I = 0; I != N; ++I) {
3135 const Expr *ArrayIndexExpr =
E->getIndexExpr(
Node.getArrayExprIndex());
3138 if (DiscardResult) {
3139 if (!this->discard(ArrayIndexExpr))
3144 if (!this->visit(ArrayIndexExpr))
3158 return this->emitOffsetOf(
T,
E,
E);
3161template <
class Emitter>
3169 if (std::optional<PrimType>
T = classify(Ty))
3170 return this->visitZeroInitializer(*
T, Ty,
E);
3174 std::optional<unsigned> LocalIndex = allocateLocal(
E);
3177 if (!this->emitGetPtrLocal(*LocalIndex,
E))
3182 QualType ElemQT = CT->getElementType();
3183 PrimType ElemT = classifyPrim(ElemQT);
3185 for (
unsigned I = 0; I != 2; ++I) {
3186 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
3188 if (!this->emitInitElem(ElemT, I,
E))
3197 std::optional<unsigned> LocalIndex = allocateLocal(
E);
3200 if (!this->emitGetPtrLocal(*LocalIndex,
E))
3205 QualType ElemQT = VT->getElementType();
3206 PrimType ElemT = classifyPrim(ElemQT);
3208 for (
unsigned I = 0, N = VT->getNumElements(); I != N; ++I) {
3209 if (!this->visitZeroInitializer(ElemT, ElemQT,
E))
3211 if (!this->emitInitElem(ElemT, I,
E))
3220template <
class Emitter>
3222 return this->emitConst(
E->getPackLength(),
E);
3225template <
class Emitter>
3228 return this->delegate(
E->getResultExpr());
3231template <
class Emitter>
3233 return this->delegate(
E->getChosenSubExpr());
3236template <
class Emitter>
3241 return this->emitConst(
E->getValue(),
E);
3244template <
class Emitter>
3249 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3250 const Function *F = this->getFunction(Ctor);
3267 if (!this->emitGetParam(PT, Offset,
E))
3272 return this->emitCall(F, 0,
E);
3275template <
class Emitter>
3279 QualType ElementType =
E->getAllocatedType();
3280 std::optional<PrimType> ElemT = classify(ElementType);
3281 unsigned PlacementArgs =
E->getNumPlacementArgs();
3283 const Expr *PlacementDest =
nullptr;
3284 bool IsNoThrow =
false;
3286 if (PlacementArgs != 0) {
3295 if (PlacementArgs == 1) {
3296 const Expr *Arg1 =
E->getPlacementArg(0);
3298 if (!this->discard(Arg1))
3303 if (!this->emitInvalidNewDeleteExpr(
E,
E))
3308 if (OperatorNew->isReservedGlobalPlacementOperator())
3309 PlacementDest = Arg1;
3313 return this->emitInvalid(
E);
3315 }
else if (!OperatorNew->isReplaceableGlobalAllocationFunction())
3316 return this->emitInvalidNewDeleteExpr(
E,
E);
3319 if (!PlacementDest) {
3328 Desc =
P.createDescriptor(
3331 false,
false,
false,
Init);
3336 std::optional<const Expr *> ArraySizeExpr =
E->getArraySize();
3340 const Expr *Stripped = *ArraySizeExpr;
3341 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3342 Stripped = ICE->getSubExpr())
3343 if (ICE->getCastKind() != CK_NoOp &&
3344 ICE->getCastKind() != CK_IntegralCast)
3349 if (PlacementDest) {
3350 if (!this->visit(PlacementDest))
3352 if (!this->visit(Stripped))
3354 if (!this->emitCheckNewTypeMismatchArray(SizeT,
E,
E))
3357 if (!this->visit(Stripped))
3362 if (!this->emitAllocN(SizeT, *ElemT,
E, IsNoThrow,
E))
3366 if (!this->emitAllocCN(SizeT, Desc, IsNoThrow,
E))
3371 if (
Init && !this->visitInitializer(
Init))
3375 if (PlacementDest) {
3376 if (!this->visit(PlacementDest))
3378 if (!this->emitCheckNewTypeMismatch(
E,
E))
3382 if (!this->emitAlloc(Desc,
E))
3388 if (!this->visit(
Init))
3391 if (!this->emitInit(*ElemT,
E))
3395 if (!this->visitInitializer(
Init))
3402 return this->emitPopPtr(
E);
3407template <
class Emitter>
3409 const Expr *Arg =
E->getArgument();
3411 const FunctionDecl *OperatorDelete =
E->getOperatorDelete();
3413 if (!OperatorDelete->isReplaceableGlobalAllocationFunction())
3414 return this->emitInvalidNewDeleteExpr(
E,
E);
3417 if (!this->visit(Arg))
3420 return this->emitFree(
E->isArrayForm(),
E->isGlobalDelete(),
E);
3423template <
class Emitter>
3434 return this->emitGetFnPtr(
Func,
E);
3437template <
class Emitter>
3441 if (!
E->isPotentiallyEvaluated()) {
3445 if (
E->isTypeOperand())
3446 return this->emitGetTypeid(
3447 E->getTypeOperand(Ctx.getASTContext()).getTypePtr(), TypeInfoType,
E);
3453 assert(
E->getExprOperand());
3454 assert(
E->getExprOperand()->
isLValue());
3456 if (!Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(
E))
3459 if (!this->visit(
E->getExprOperand()))
3462 if (!this->emitGetTypeidPtr(TypeInfoType,
E))
3465 return this->emitPopPtr(
E);
3469template <
class Emitter>
3471 assert(Ctx.getLangOpts().CPlusPlus);
3472 return this->emitConstBool(
E->getValue(),
E);
3475template <
class Emitter>
3487 return this->emitDummyPtr(GuidDecl,
E);
3489 std::optional<unsigned> GlobalIndex =
P.getOrCreateGlobal(GuidDecl);
3492 if (!this->emitGetPtrGlobal(*GlobalIndex,
E))
3495 assert(this->getRecord(
E->
getType()));
3501 assert(
V.isStruct());
3502 assert(
V.getStructNumBases() == 0);
3503 if (!this->visitAPValueInitializer(
V,
E))
3506 return this->emitFinishInit(
E);
3509template <
class Emitter>
3514 return this->emitConstBool(
E->isSatisfied(),
E);
3517template <
class Emitter>
3523 return this->emitConstBool(
E->isSatisfied(),
E);
3526template <
class Emitter>
3529 return this->delegate(
E->getSemanticForm());
3532template <
class Emitter>
3535 for (
const Expr *SemE :
E->semantics()) {
3536 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
3537 if (SemE ==
E->getResultExpr())
3540 if (OVE->isUnique())
3543 if (!this->discard(OVE))
3545 }
else if (SemE ==
E->getResultExpr()) {
3546 if (!this->delegate(SemE))
3549 if (!this->discard(SemE))
3556template <
class Emitter>
3558 return this->delegate(
E->getSelectedExpr());
3561template <
class Emitter>
3563 return this->emitError(
E);
3566template <
class Emitter>
3570 unsigned Offset = allocateLocalPrimitive(
3571 E->getLabel(),
PT_Ptr,
true,
false);
3573 return this->emitGetLocal(
PT_Ptr, Offset,
E);
3576template <
class Emitter>
3580 QualType ElemType = VT->getElementType();
3581 PrimType ElemT = classifyPrim(ElemType);
3582 const Expr *Src =
E->getSrcExpr();
3584 PrimType SrcElemT = classifyVectorElementType(SrcType);
3586 unsigned SrcOffset = this->allocateLocalPrimitive(Src,
PT_Ptr,
true,
false);
3587 if (!this->visit(Src))
3589 if (!this->emitSetLocal(
PT_Ptr, SrcOffset,
E))
3592 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
3593 if (!this->emitGetLocal(
PT_Ptr, SrcOffset,
E))
3595 if (!this->emitArrayElemPop(SrcElemT, I,
E))
3599 if (SrcElemT != ElemT) {
3600 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType,
E))
3602 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
3603 const auto *TargetSemantics = &Ctx.getFloatSemantics(ElemType);
3607 if (!this->emitInitElem(ElemT, I,
E))
3614template <
class Emitter>
3617 assert(
E->getNumSubExprs() > 2);
3619 const Expr *Vecs[] = {
E->getExpr(0),
E->getExpr(1)};
3623 unsigned NumOutputElems =
E->getNumSubExprs() - 2;
3624 assert(NumOutputElems > 0);
3627 unsigned VectorOffsets[2];
3628 for (
unsigned I = 0; I != 2; ++I) {
3629 VectorOffsets[I] = this->allocateLocalPrimitive(
3630 Vecs[I],
PT_Ptr,
true,
false);
3631 if (!this->visit(Vecs[I]))
3633 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I],
E))
3636 for (
unsigned I = 0; I != NumOutputElems; ++I) {
3637 APSInt ShuffleIndex =
E->getShuffleMaskIdx(Ctx.getASTContext(), I);
3638 assert(ShuffleIndex >= -1);
3639 if (ShuffleIndex == -1)
3640 return this->emitInvalidShuffleVectorIndex(I,
E);
3642 assert(ShuffleIndex < (NumInputElems * 2));
3643 if (!this->emitGetLocal(
PT_Ptr,
3644 VectorOffsets[ShuffleIndex >= NumInputElems],
E))
3646 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
3647 if (!this->emitArrayElemPop(ElemT, InputVectorIndex,
E))
3650 if (!this->emitInitElem(ElemT, I,
E))
3657template <
class Emitter>
3662 Base->getType()->isVectorType() ||
3666 E->getEncodedElementAccess(Indices);
3668 if (Indices.size() == 1) {
3669 if (!this->visit(
Base))
3673 if (!this->emitConstUint32(Indices[0],
E))
3675 return this->emitArrayElemPtrPop(
PT_Uint32,
E);
3678 return this->emitArrayElemPop(classifyPrim(
E->
getType()), Indices[0],
E);
3682 unsigned BaseOffset = allocateLocalPrimitive(
Base,
PT_Ptr,
true,
3684 if (!this->visit(
Base))
3686 if (!this->emitSetLocal(
PT_Ptr, BaseOffset,
E))
3691 std::optional<unsigned> ResultIndex;
3692 ResultIndex = allocateLocal(
E);
3695 if (!this->emitGetPtrLocal(*ResultIndex,
E))
3703 uint32_t DstIndex = 0;
3704 for (uint32_t I : Indices) {
3705 if (!this->emitGetLocal(
PT_Ptr, BaseOffset,
E))
3707 if (!this->emitArrayElemPop(ElemT, I,
E))
3709 if (!this->emitInitElem(ElemT, DstIndex,
E))
3715 assert(!DiscardResult);
3719template <
class Emitter>
3721 const Expr *SubExpr =
E->getSubExpr();
3722 if (!
E->isExpressibleAsConstantInitializer())
3723 return this->discard(SubExpr) && this->emitInvalid(
E);
3728 assert(classifyPrim(
E) ==
PT_Ptr);
3729 return this->emitDummyPtr(
E,
E);
3732template <
class Emitter>
3735 const Expr *SubExpr =
E->getSubExpr();
3737 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
3742 if (!this->visit(SubExpr))
3744 if (!this->emitConstUint8(0,
E))
3746 if (!this->emitArrayElemPtrPopUint8(
E))
3758 assert(SecondFieldT ==
PT_Ptr);
3762 if (!this->emitExpandPtr(
E))
3766 if (!this->emitArrayElemPtrPop(
PT_Uint64,
E))
3771template <
class Emitter>
3780 if (!this->visitStmt(S))
3785 assert(S == Result);
3786 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
3787 return this->delegate(ResultExpr);
3788 return this->emitUnsupported(
E);
3797 return this->Visit(
E);
3804 return this->Visit(
E);
3812 return this->discard(
E);
3817 std::optional<unsigned> LocalIndex = allocateLocal(
E);
3821 if (!this->emitGetPtrLocal(*LocalIndex,
E))
3824 return this->visitInitializer(
E);
3831 return this->Visit(
E);
3834template <
class Emitter>
3840 return this->Visit(
E);
3844 std::optional<PrimType>
T = classify(
E->
getType());
3848 if (!this->visit(
E))
3850 return this->emitComplexBoolCast(
E);
3855 if (!this->visit(
E))
3863 if (!this->emitNull(*
T, 0,
nullptr,
E))
3865 return this->emitNE(*
T,
E);
3870 return this->emitCastFloatingIntegralBool(getFPOptions(
E),
E);
3876template <
class Emitter>
3881 return this->emitZeroBool(
E);
3883 return this->emitZeroSint8(
E);
3885 return this->emitZeroUint8(
E);
3887 return this->emitZeroSint16(
E);
3889 return this->emitZeroUint16(
E);
3891 return this->emitZeroSint32(
E);
3893 return this->emitZeroUint32(
E);
3895 return this->emitZeroSint64(
E);
3897 return this->emitZeroUint64(
E);
3899 return this->emitZeroIntAP(Ctx.getBitWidth(QT),
E);
3901 return this->emitZeroIntAPS(Ctx.getBitWidth(QT),
E);
3903 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
3906 return this->emitNullFnPtr(0,
nullptr,
E);
3908 return this->emitNullMemberPtr(0,
nullptr,
E);
3910 return this->emitConstFloat(APFloat::getZero(Ctx.getFloatSemantics(QT)),
E);
3912 auto Sem = Ctx.getASTContext().getFixedPointSemantics(
E->
getType());
3915 llvm_unreachable(
"Implement");
3917 llvm_unreachable(
"unknown primitive type");
3920template <
class Emitter>
3926 for (
const Record::Field &Field : R->
fields()) {
3927 if (
Field.Decl->isUnnamedBitField())
3931 if (
D->isPrimitive()) {
3934 if (!this->visitZeroInitializer(
T, QT,
E))
3936 if (!this->emitInitField(
T,
Field.Offset,
E))
3943 if (!this->emitGetPtrField(
Field.Offset,
E))
3946 if (
D->isPrimitiveArray()) {
3949 for (uint32_t I = 0, N =
D->getNumElems(); I != N; ++I) {
3950 if (!this->visitZeroInitializer(
T, ET,
E))
3952 if (!this->emitInitElem(
T, I,
E))
3955 }
else if (
D->isCompositeArray()) {
3957 if (!this->visitZeroArrayInitializer(
D->getType(),
E))
3959 }
else if (
D->isRecord()) {
3960 if (!this->visitZeroRecordInitializer(
D->ElemRecord,
E))
3966 if (!this->emitFinishInitPop(
E))
3975 for (
const Record::Base &B : R->
bases()) {
3976 if (!this->emitGetPtrBase(B.Offset,
E))
3978 if (!this->visitZeroRecordInitializer(B.R,
E))
3980 if (!this->emitFinishInitPop(
E))
3989template <
class Emitter>
3994 size_t NumElems = cast<ConstantArrayType>(AT)->getZExtSize();
3996 if (std::optional<PrimType> ElemT = classify(ElemType)) {
3997 for (
size_t I = 0; I != NumElems; ++I) {
3998 if (!this->visitZeroInitializer(*ElemT, ElemType,
E))
4000 if (!this->emitInitElem(*ElemT, I,
E))
4005 const Record *R = getRecord(ElemType);
4007 for (
size_t I = 0; I != NumElems; ++I) {
4008 if (!this->emitConstUint32(I,
E))
4012 if (!this->visitZeroRecordInitializer(R,
E))
4014 if (!this->emitPopPtr(
E))
4019 for (
size_t I = 0; I != NumElems; ++I) {
4020 if (!this->emitConstUint32(I,
E))
4024 if (!this->visitZeroArrayInitializer(ElemType,
E))
4026 if (!this->emitPopPtr(
E))
4035template <
class Emitter>
4036template <
typename T>
4040 return this->emitConstSint8(
Value,
E);
4042 return this->emitConstUint8(
Value,
E);
4044 return this->emitConstSint16(
Value,
E);
4046 return this->emitConstUint16(
Value,
E);
4048 return this->emitConstSint32(
Value,
E);
4050 return this->emitConstUint32(
Value,
E);
4052 return this->emitConstSint64(
Value,
E);
4054 return this->emitConstUint64(
Value,
E);
4056 return this->emitConstBool(
Value,
E);
4064 llvm_unreachable(
"Invalid integral type");
4067 llvm_unreachable(
"unknown primitive type");
4070template <
class Emitter>
4071template <
typename T>
4076template <
class Emitter>
4080 return this->emitConstIntAPS(
Value,
E);
4082 return this->emitConstIntAP(
Value,
E);
4084 if (
Value.isSigned())
4085 return this->emitConst(
Value.getSExtValue(), Ty,
E);
4086 return this->emitConst(
Value.getZExtValue(), Ty,
E);
4089template <
class Emitter>
4094template <
class Emitter>
4099 if (
const auto *VD =
4100 dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
4101 assert(!
P.getGlobal(VD));
4102 assert(!Locals.contains(VD));
4110 isa<const Expr *>(Src));
4112 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
4113 Locals.insert({VD, Local});
4114 VarScope->add(Local, IsExtended);
4115 return Local.Offset;
4118template <
class Emitter>
4119std::optional<unsigned>
4123 if ([[maybe_unused]]
const auto *VD =
4124 dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
4125 assert(!
P.getGlobal(VD));
4126 assert(!Locals.contains(VD));
4131 bool IsTemporary =
false;
4132 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
4136 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
4137 Init = VarD->getInit();
4139 if (
auto *
E = Src.dyn_cast<
const Expr *>()) {
4147 IsTemporary,
false,
Init);
4149 return std::nullopt;
4153 Locals.insert({Key, Local});
4155 VarScope->addExtended(Local, ExtendingDecl);
4157 VarScope->add(Local,
false);
4158 return Local.Offset;
4161template <
class Emitter>
4168 true,
false,
nullptr);
4175 while (S->getParent())
4177 assert(S && !S->getParent());
4179 return Local.Offset;
4182template <
class Emitter>
4184 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
4190 if (
const auto *RecordTy = getRecordTy(Ty))
4191 return getRecord(RecordTy->getDecl());
4195template <
class Emitter>
4197 return P.getOrCreateRecord(RD);
4200template <
class Emitter>
4202 return Ctx.getOrCreateFunction(FD);
4205template <
class Emitter>
4210 if (!DestroyToplevelScope) {
4211 if (!this->emitCheckAllocations(
E))
4215 auto maybeDestroyLocals = [&]() ->
bool {
4216 if (DestroyToplevelScope)
4217 return RootScope.
destroyLocals() && this->emitCheckAllocations(
E);
4218 return this->emitCheckAllocations(
E);
4225 return this->emitRetVoid(
E) && maybeDestroyLocals();
4229 if (std::optional<PrimType>
T = classify(
E)) {
4233 return this->emitRet(*
T,
E) && maybeDestroyLocals();
4239 if (std::optional<unsigned> LocalOffset = this->allocateLocal(
E)) {
4240 if (!this->emitGetPtrLocal(*LocalOffset,
E))
4243 if (!visitInitializer(
E))
4246 if (!this->emitFinishInit(
E))
4251 return this->emitRetValue(
E) && maybeDestroyLocals();
4254 return maybeDestroyLocals() && this->emitCheckAllocations(
E) &&
false;
4257template <
class Emitter>
4260 auto R = this->visitVarDecl(VD,
true);
4269 if (
auto GlobalIndex =
P.getGlobal(VD)) {
4270 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
4274 GD.
InitState = GlobalInitState::InitializerFailed;
4285template <
class Emitter>
4287 bool ConstantContext) {
4288 std::optional<PrimType> VarT = classify(VD->
getType());
4292 if (!ConstantContext) {
4297 this->emitCheckAllocations(VD);
4301 if (!this->visitVarDecl(VD,
true))
4305 auto GlobalIndex =
P.getGlobal(VD);
4306 assert(GlobalIndex);
4308 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
4311 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
4315 auto Local = Locals.find(VD);
4316 assert(Local != Locals.end());
4318 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
4321 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
4327 if (!this->emitRet(VarT.value_or(
PT_Ptr), VD)) {
4331 auto GlobalIndex =
P.getGlobal(VD);
4332 assert(GlobalIndex);
4333 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
4337 GD.
InitState = GlobalInitState::InitializerFailed;
4343 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
4346template <
class Emitter>
4355 if (!this->isActive())
4359 std::optional<PrimType> VarT = classify(VD->
getType());
4361 if (
Init &&
Init->isValueDependent())
4365 auto checkDecl = [&]() ->
bool {
4367 return !NeedsOp || this->emitCheckDecl(VD, VD);
4370 auto initGlobal = [&](
unsigned GlobalIndex) ->
bool {
4374 if (!this->visit(
Init))
4375 return checkDecl() &&
false;
4377 return checkDecl() && this->emitInitGlobal(*VarT, GlobalIndex, VD);
4383 if (!this->emitGetPtrGlobal(GlobalIndex,
Init))
4386 if (!visitInitializer(
Init))
4389 if (!this->emitFinishInit(
Init))
4392 return this->emitPopPtr(
Init);
4398 if (std::optional<unsigned> GlobalIndex =
P.getGlobal(VD)) {
4399 if (
P.getPtrGlobal(*GlobalIndex).isInitialized())
4404 return Init && checkDecl() && initGlobal(*GlobalIndex);
4407 std::optional<unsigned> GlobalIndex =
P.createGlobal(VD,
Init);
4412 return !
Init || (checkDecl() && initGlobal(*GlobalIndex));
4417 unsigned Offset = this->allocateLocalPrimitive(
4424 if (!this->visit(
Init))
4426 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
4428 if (!this->visit(
Init))
4430 return this->emitSetLocal(*VarT, Offset, VD);
4434 if (std::optional<unsigned> Offset = this->allocateLocal(VD)) {
4438 if (!this->emitGetPtrLocal(*Offset,
Init))
4441 if (!visitInitializer(
Init))
4444 if (!this->emitFinishInit(
Init))
4447 return this->emitPopPtr(
Init);
4457template <
class Emitter>
4460 assert(!DiscardResult);
4462 return this->emitConst(Val.
getInt(), ValType,
E);
4464 return this->emitConstFloat(Val.
getFloat(),
E);
4468 return this->emitNull(ValType, 0,
nullptr,
E);
4470 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
4471 return this->visit(BaseExpr);
4472 else if (
const auto *VD =
Base.dyn_cast<
const ValueDecl *>()) {
4473 return this->visitDeclRef(VD,
E);
4477 return this->emitGetMemberPtr(MemberDecl,
E);
4478 return this->emitNullMemberPtr(0,
nullptr,
E);
4484template <
class Emitter>
4493 const Record::Field *RF = R->
getField(I);
4496 PrimType T = classifyPrim(RF->Decl->getType());
4497 if (!this->visitAPValue(F,
T,
E))
4499 if (!this->emitInitField(
T, RF->Offset,
E))
4502 assert(RF->Desc->isPrimitiveArray());
4503 const auto *ArrType = RF->Decl->getType()->getAsArrayTypeUnsafe();
4504 PrimType ElemT = classifyPrim(ArrType->getElementType());
4507 if (!this->emitGetPtrField(RF->Offset,
E))
4510 for (
unsigned A = 0, AN = F.
getArraySize(); A != AN; ++A) {
4513 if (!this->emitInitElem(ElemT, A,
E))
4517 if (!this->emitPopPtr(
E))
4520 if (!this->emitGetPtrField(RF->Offset,
E))
4522 if (!this->visitAPValueInitializer(F,
E))
4524 if (!this->emitPopPtr(
E))
4527 assert(
false &&
"I don't think this should be possible");
4536 const Record::Field *RF = R->
getField(UnionField);
4537 PrimType T = classifyPrim(RF->Decl->getType());
4538 if (!this->visitAPValue(F,
T,
E))
4540 return this->emitInitField(
T, RF->Offset,
E);
4547template <
class Emitter>
4549 unsigned BuiltinID) {
4556 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
4557 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
4558 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
4559 BuiltinID == Builtin::BI__builtin_function_start) {
4562 return this->emitDummyPtr(
E,
E);
4566 std::optional<PrimType> ReturnT = classify(
E);
4570 std::optional<unsigned> LocalIndex = allocateLocal(
E);
4573 if (!this->emitGetPtrLocal(*LocalIndex,
E))
4577 if (!
Func->isUnevaluatedBuiltin()) {
4579 for (
const auto *Arg :
E->arguments()) {
4580 if (!this->visit(Arg))
4585 if (!this->emitCallBI(
Func,
E, BuiltinID,
E))
4588 if (DiscardResult && !ReturnType->
isVoidType()) {
4590 return this->emitPop(*ReturnT,
E);
4596template <
class Emitter>
4598 if (
unsigned BuiltinID =
E->getBuiltinCallee())
4599 return VisitBuiltinCallExpr(
E, BuiltinID);
4606 return VisitBuiltinCallExpr(
E, Builtin::BI__builtin_operator_new);
4609 return VisitBuiltinCallExpr(
E, Builtin::BI__builtin_operator_delete);
4613 if (
const auto *DD = dyn_cast_if_present<CXXDestructorDecl>(FuncDecl);
4614 DD && DD->isTrivial())
4617 QualType ReturnType =
E->getCallReturnType(Ctx.getASTContext());
4618 std::optional<PrimType>
T = classify(ReturnType);
4622 if (DiscardResult) {
4626 if (std::optional<unsigned> LocalIndex = allocateLocal(
E)) {
4627 if (!this->emitGetPtrLocal(*LocalIndex,
E))
4634 if (std::optional<unsigned> LocalIndex = allocateLocal(
E)) {
4635 if (!this->emitGetPtrLocal(*LocalIndex,
E))
4639 if (!this->emitDupPtr(
E))
4647 bool IsAssignmentOperatorCall =
false;
4648 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(
E);
4649 OCE && OCE->isAssignmentOp()) {
4653 assert(Args.size() == 2);
4654 IsAssignmentOperatorCall =
true;
4655 std::reverse(Args.begin(), Args.end());
4660 if (isa<CXXOperatorCallExpr>(
E)) {
4661 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
4662 MD && MD->isStatic()) {
4663 if (!this->discard(
E->getArg(0)))
4666 Args.erase(Args.begin());
4670 std::optional<unsigned> CalleeOffset;
4672 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(
E)) {
4673 if (!FuncDecl && classifyPrim(
E->getCallee()) ==
PT_MemberPtr) {
4677 const Expr *Callee =
E->getCallee();
4679 this->allocateLocalPrimitive(Callee,
PT_MemberPtr,
true,
false);
4680 if (!this->visit(Callee))
4686 if (!this->emitGetMemberPtrBase(
E))
4688 }
else if (!this->visit(MC->getImplicitObjectArgument())) {
4691 }
else if (!FuncDecl) {
4692 const Expr *Callee =
E->getCallee();
4693 CalleeOffset = this->allocateLocalPrimitive(Callee,
PT_FnPtr,
true,
false);
4694 if (!this->visit(Callee))
4696 if (!this->emitSetLocal(
PT_FnPtr, *CalleeOffset,
E))
4702 unsigned ArgIndex = 0;
4703 for (
const auto *Arg : Args) {
4704 if (!this->visit(Arg))
4708 if (FuncDecl && NonNullArgs[ArgIndex]) {
4711 if (!this->emitCheckNonNullArg(ArgT, Arg))
4719 if (IsAssignmentOperatorCall) {
4720 assert(Args.size() == 2);
4723 if (!this->emitFlip(Arg2T, Arg1T,
E))
4731 assert(HasRVO ==
Func->hasRVO());
4733 bool HasQualifier =
false;
4734 if (
const auto *ME = dyn_cast<MemberExpr>(
E->getCallee()))
4735 HasQualifier = ME->hasQualifier();
4737 bool IsVirtual =
false;
4738 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
4739 IsVirtual = MD->isVirtual();
4744 if (IsVirtual && !HasQualifier) {
4745 uint32_t VarArgSize = 0;
4746 unsigned NumParams =
4747 Func->getNumWrittenParams() + isa<CXXOperatorCallExpr>(
E);
4748 for (
unsigned I = NumParams, N =
E->getNumArgs(); I != N; ++I)
4751 if (!this->emitCallVirt(
Func, VarArgSize,
E))
4753 }
else if (
Func->isVariadic()) {
4754 uint32_t VarArgSize = 0;
4755 unsigned NumParams =
4756 Func->getNumWrittenParams() + isa<CXXOperatorCallExpr>(
E);
4757 for (
unsigned I = NumParams, N =
E->getNumArgs(); I != N; ++I)
4759 if (!this->emitCallVar(
Func, VarArgSize,
E))
4762 if (!this->emitCall(
Func, 0,
E))
4771 uint32_t ArgSize = 0;
4772 for (
unsigned I = 0, N =
E->getNumArgs(); I != N; ++I)
4777 if (isa<CXXMemberCallExpr>(
E) && CalleeOffset) {
4780 if (!this->emitGetMemberPtrDecl(
E))
4783 if (!this->emitGetLocal(
PT_FnPtr, *CalleeOffset,
E))
4786 if (!this->emitCallPtr(ArgSize,
E,
E))
4791 if (DiscardResult && !ReturnType->
isVoidType() &&
T)
4792 return this->emitPop(*
T,
E);
4797template <
class Emitter>
4801 return this->delegate(
E->getExpr());
4804template <
class Emitter>
4808 const Expr *SubExpr =
E->getExpr();
4809 if (std::optional<PrimType>
T = classify(
E->getExpr()))
4810 return this->visit(SubExpr);
4813 return this->visitInitializer(SubExpr);
4816template <
class Emitter>
4821 return this->emitConstBool(
E->getValue(),
E);
4824template <
class Emitter>
4830 uint64_t Val = Ctx.getASTContext().getTargetNullPointerValue(
E->
getType());
4831 return this->emitNullPtr(Val,
nullptr,
E);
4834template <
class Emitter>
4842 return this->emitZero(
T,
E);
4845template <
class Emitter>
4850 if (this->LambdaThisCapture.Offset > 0) {
4851 if (this->LambdaThisCapture.IsPtr)
4852 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset,
E);
4853 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset,
E);
4860 if (!InitStackActive)
4861 return this->emitThis(
E);
4863 if (!InitStack.empty()) {
4877 unsigned StartIndex = 0;
4878 unsigned EndIndex = 0;
4880 for (StartIndex = InitStack.size() - 1; StartIndex > 0; --StartIndex) {
4883 EndIndex = StartIndex;
4890 for (; StartIndex > 0; --StartIndex) {
4900 for (
unsigned I = StartIndex; I != EndIndex; ++I) {
4903 if (!InitStack[I].
template emit<Emitter>(
this,
E))
4908 return this->emitThis(
E);
4912 switch (S->getStmtClass()) {
4913 case Stmt::CompoundStmtClass:
4914 return visitCompoundStmt(cast<CompoundStmt>(S));
4915 case Stmt::DeclStmtClass:
4916 return visitDeclStmt(cast<DeclStmt>(S));
4917 case Stmt::ReturnStmtClass:
4918 return visitReturnStmt(cast<ReturnStmt>(S));
4919 case Stmt::IfStmtClass:
4920 return visitIfStmt(cast<IfStmt>(S));
4921 case Stmt::WhileStmtClass:
4922 return visitWhileStmt(cast<WhileStmt>(S));
4923 case Stmt::DoStmtClass:
4924 return visitDoStmt(cast<DoStmt>(S));
4925 case Stmt::ForStmtClass:
4926 return visitForStmt(cast<ForStmt>(S));
4927 case Stmt::CXXForRangeStmtClass:
4928 return visitCXXForRangeStmt(cast<CXXForRangeStmt>(S));
4929 case Stmt::BreakStmtClass:
4930 return visitBreakStmt(cast<BreakStmt>(S));
4931 case Stmt::ContinueStmtClass:
4932 return visitContinueStmt(cast<ContinueStmt>(S));
4933 case Stmt::SwitchStmtClass:
4934 return visitSwitchStmt(cast<SwitchStmt>(S));
4935 case Stmt::CaseStmtClass:
4936 return visitCaseStmt(cast<CaseStmt>(S));
4937 case Stmt::DefaultStmtClass:
4938 return visitDefaultStmt(cast<DefaultStmt>(S));
4939 case Stmt::AttributedStmtClass:
4940 return visitAttributedStmt(cast<AttributedStmt>(S));
4941 case Stmt::CXXTryStmtClass:
4942 return visitCXXTryStmt(cast<CXXTryStmt>(S));
4943 case Stmt::NullStmtClass:
4946 case Stmt::GCCAsmStmtClass:
4947 case Stmt::MSAsmStmtClass:
4948 case Stmt::GotoStmtClass:
4949 return this->emitInvalid(S);
4950 case Stmt::LabelStmtClass:
4951 return this->visitStmt(cast<LabelStmt>(S)->getSubStmt());
4953 if (
const auto *
E = dyn_cast<Expr>(S))
4954 return this->discard(
E);
4960template <
class Emitter>
4963 for (
const auto *InnerStmt : S->body())
4964 if (!visitStmt(InnerStmt))
4966 return Scope.destroyLocals();
4969template <
class Emitter>
4971 for (
const auto *
D : DS->
decls()) {
4976 const auto *VD = dyn_cast<VarDecl>(
D);
4979 if (!this->visitVarDecl(VD))
4986template <
class Emitter>
4988 if (this->InStmtExpr)
4989 return this->emitUnsupported(RS);
4995 if (!this->visit(RE))
4997 this->emitCleanup();
4998 return this->emitRet(*ReturnType, RS);
4999 }
else if (RE->getType()->isVoidType()) {
5000 if (!this->visit(RE))
5005 if (!this->emitRVOPtr(RE))
5007 if (!this->visitInitializer(RE))
5009 if (!this->emitPopPtr(RE))
5012 this->emitCleanup();
5013 return this->emitRetVoid(RS);
5018 this->emitCleanup();
5019 return this->emitRetVoid(RS);
5023 if (
auto *CondInit = IS->
getInit())
5024 if (!visitStmt(CondInit))
5028 if (!visitDeclStmt(CondDecl))
5033 if (!this->emitIsConstantContext(IS))
5036 if (!this->emitIsConstantContext(IS))
5038 if (!this->emitInv(IS))
5041 if (!this->visitBool(IS->
getCond()))
5046 LabelTy LabelElse = this->getLabel();
5047 LabelTy LabelEnd = this->getLabel();
5048 if (!this->jumpFalse(LabelElse))
5052 if (!visitStmt(IS->
getThen()))
5057 if (!this->jump(LabelEnd))
5059 this->emitLabel(LabelElse);
5062 if (!visitStmt(Else))
5067 this->emitLabel(LabelEnd);
5069 LabelTy LabelEnd = this->getLabel();
5070 if (!this->jumpFalse(LabelEnd))
5074 if (!visitStmt(IS->
getThen()))
5079 this->emitLabel(LabelEnd);
5085template <
class Emitter>
5087 const Expr *Cond = S->getCond();
5088 const Stmt *Body = S->getBody();
5090 LabelTy CondLabel = this->getLabel();
5091 LabelTy EndLabel = this->getLabel();
5094 this->fallthrough(CondLabel);
5095 this->emitLabel(CondLabel);
5099 if (
const DeclStmt *CondDecl = S->getConditionVariableDeclStmt())
5100 if (!visitDeclStmt(CondDecl))
5103 if (!this->visitBool(Cond))
5105 if (!this->jumpFalse(EndLabel))
5108 if (!this->visitStmt(Body))
5114 if (!this->jump(CondLabel))
5116 this->fallthrough(EndLabel);
5117 this->emitLabel(EndLabel);
5123 const Expr *Cond = S->getCond();
5124 const Stmt *Body = S->getBody();
5126 LabelTy StartLabel = this->getLabel();
5127 LabelTy EndLabel = this->getLabel();
5128 LabelTy CondLabel = this->getLabel();
5131 this->fallthrough(StartLabel);
5132 this->emitLabel(StartLabel);
5136 if (!this->visitStmt(Body))
5138 this->fallthrough(CondLabel);
5139 this->emitLabel(CondLabel);
5140 if (!this->visitBool(Cond))
5146 if (!this->jumpTrue(StartLabel))
5149 this->fallthrough(EndLabel);
5150 this->emitLabel(EndLabel);
5154template <
class Emitter>
5158 const Expr *Cond = S->getCond();
5159 const Expr *
Inc = S->getInc();
5160 const Stmt *Body = S->getBody();
5162 LabelTy EndLabel = this->getLabel();
5163 LabelTy CondLabel = this->getLabel();
5164 LabelTy IncLabel = this->getLabel();
5167 if (
Init && !this->visitStmt(
Init))
5170 this->fallthrough(CondLabel);
5171 this->emitLabel(CondLabel);
5175 if (
const DeclStmt *CondDecl = S->getConditionVariableDeclStmt())
5176 if (!visitDeclStmt(CondDecl))
5180 if (!this->visitBool(Cond))
5182 if (!this->jumpFalse(EndLabel))
5186 if (Body && !this->visitStmt(Body))
5189 this->fallthrough(IncLabel);
5190 this->emitLabel(IncLabel);
5191 if (
Inc && !this->discard(
Inc))
5197 if (!this->jump(CondLabel))
5200 this->fallthrough(EndLabel);
5201 this->emitLabel(EndLabel);
5205template <
class Emitter>
5208 const Expr *Cond = S->getCond();
5209 const Expr *
Inc = S->getInc();
5210 const Stmt *Body = S->getBody();
5211 const Stmt *BeginStmt = S->getBeginStmt();
5212 const Stmt *RangeStmt = S->getRangeStmt();
5213 const Stmt *EndStmt = S->getEndStmt();
5214 const VarDecl *LoopVar = S->getLoopVariable();
5216 LabelTy EndLabel = this->getLabel();
5217 LabelTy CondLabel = this->getLabel();
5218 LabelTy IncLabel = this->getLabel();
5222 if (
Init && !this->visitStmt(
Init))
5224 if (!this->visitStmt(RangeStmt))
5226 if (!this->visitStmt(BeginStmt))
5228 if (!this->visitStmt(EndStmt))
5232 this->fallthrough(CondLabel);
5233 this->emitLabel(CondLabel);
5234 if (!this->visitBool(Cond))
5236 if (!this->jumpFalse(EndLabel))
5239 if (!this->visitVarDecl(LoopVar))
5244 if (!this->visitStmt(Body))
5247 this->fallthrough(IncLabel);
5248 this->emitLabel(IncLabel);
5249 if (!this->discard(
Inc))
5253 if (!this->jump(CondLabel))
5256 this->fallthrough(EndLabel);
5257 this->emitLabel(EndLabel);
5261template <
class Emitter>
5268 C->emitDestruction();
5269 return this->jump(*BreakLabel);
5272template <
class Emitter>
5278 C &&
C->getParent() != ContinueVarScope;
C =
C->getParent())
5279 C->emitDestruction();
5280 return this->jump(*ContinueLabel);
5283template <
class Emitter>
5285 const Expr *Cond = S->getCond();
5289 LabelTy EndLabel = this->getLabel();
5291 unsigned CondVar = this->allocateLocalPrimitive(Cond, CondT,
true,
false);
5293 if (
const auto *CondInit = S->getInit())
5294 if (!visitStmt(CondInit))
5297 if (
const DeclStmt *CondDecl = S->getConditionVariableDeclStmt())
5298 if (!visitDeclStmt(CondDecl))
5302 if (!this->visit(Cond))
5304 if (!this->emitSetLocal(CondT, CondVar, S))
5309 for (
const SwitchCase *SC = S->getSwitchCaseList(); SC;
5310 SC = SC->getNextSwitchCase()) {
5311 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
5313 if (CS->caseStmtIsGNURange())
5315 CaseLabels[SC] = this->getLabel();
5321 if (!this->emitGetLocal(CondT, CondVar, CS))
5323 if (!this->visit(
Value))
5327 if (!this->emitEQ(ValueT, S))
5329 if (!this->jumpTrue(CaseLabels[CS]))
5332 assert(!DefaultLabel);
5333 DefaultLabel = this->getLabel();
5340 if (!this->jump(*DefaultLabel))
5343 if (!this->jump(EndLabel))
5348 if (!this->visitStmt(S->getBody()))
5350 this->emitLabel(EndLabel);
5355template <
class Emitter>
5357 this->emitLabel(CaseLabels[S]);
5358 return this->visitStmt(S->getSubStmt());
5361template <
class Emitter>
5363 this->emitLabel(*DefaultLabel);
5364 return this->visitStmt(S->getSubStmt());
5367template <
class Emitter>
5369 if (this->Ctx.getLangOpts().CXXAssumptions &&
5370 !this->Ctx.getLangOpts().MSVCCompat) {
5371 for (
const Attr *A : S->getAttrs()) {
5372 auto *AA = dyn_cast<CXXAssumeAttr>(A);
5376 assert(isa<NullStmt>(S->getSubStmt()));
5378 const Expr *Assumption = AA->getAssumption();
5386 if (!this->visitBool(Assumption))
5389 if (!this->emitAssume(Assumption))
5395 return this->visitStmt(S->getSubStmt());
5398template <
class Emitter>
5401 return this->visitStmt(S->getTryBlock());
5404template <
class Emitter>
5408 assert(cast<CompoundStmt>(MD->
getBody())->body_empty());
5413 const Function *
Func = this->getFunction(LambdaCallOp);
5416 assert(
Func->hasThisPointer());
5419 if (
Func->hasRVO()) {
5420 if (!this->emitRVOPtr(MD))
5428 if (!this->emitNullPtr(0,
nullptr, MD))
5433 auto It = this->Params.find(PVD);
5434 assert(It != this->Params.end());
5438 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
5439 if (!this->emitGetParam(ParamType, It->second.Offset, MD))
5443 if (!this->emitCall(
Func, 0, LambdaCallOp))
5446 this->emitCleanup();
5448 return this->emitRet(*ReturnType, MD);
5451 return this->emitRetVoid(MD);
5454template <
class Emitter>
5456 if (Ctx.getLangOpts().CPlusPlus23)
5465template <
class Emitter>
5467 assert(!ReturnType);
5469 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
5470 const Expr *InitExpr) ->
bool {
5472 if (InitExpr->getType().isNull())
5475 if (std::optional<PrimType>
T = this->classify(InitExpr)) {
5476 if (!this->visit(InitExpr))
5479 if (F->isBitField())
5480 return this->emitInitThisBitField(*
T, F, FieldOffset, InitExpr);
5481 return this->emitInitThisField(*
T, FieldOffset, InitExpr);
5486 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
5489 if (!this->visitInitializer(InitExpr))
5492 return this->emitFinishInitPop(InitExpr);
5496 const Record *R = this->getRecord(RD);
5502 assert(cast<CompoundStmt>(Ctor->
getBody())->body_empty());
5503 if (!this->emitThis(Ctor))
5512 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
5513 this->emitRetVoid(Ctor);
5517 for (
const auto *
Init : Ctor->
inits()) {
5521 const Expr *InitExpr =
Init->getInit();
5525 if (!emitFieldInitializer(F, F->Offset, InitExpr))
5528 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
5531 if (
Init->isBaseVirtual()) {
5533 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
5539 const Record::Base *B = R->
getBase(BaseDecl);
5541 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
5545 if (!this->visitInitializer(InitExpr))
5547 if (!this->emitFinishInitPop(InitExpr))
5550 assert(IFD->getChainingSize() >= 2);
5552 unsigned NestedFieldOffset = 0;
5553 const Record::Field *NestedField =
nullptr;
5554 for (
const NamedDecl *ND : IFD->chain()) {
5555 const auto *FD = cast<FieldDecl>(ND);
5556 const Record *FieldRecord = this->
P.getOrCreateRecord(FD->getParent());
5557 assert(FieldRecord);
5559 NestedField = FieldRecord->
getField(FD);
5560 assert(NestedField);
5562 NestedFieldOffset += NestedField->Offset;
5564 assert(NestedField);
5566 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr))
5569 assert(
Init->isDelegatingInitializer());
5570 if (!this->emitThis(InitExpr))
5572 if (!this->visitInitializer(
Init->getInit()))
5574 if (!this->emitPopPtr(InitExpr))
5578 if (!
Scope.destroyLocals())
5582 if (
const auto *Body = Ctor->
getBody())
5583 if (!visitStmt(Body))
5589template <
class Emitter>
5592 const Record *R = this->getRecord(RD);
5597 if (!this->visitStmt(Dtor->
getBody()))
5601 if (!this->emitThis(Dtor))
5607 for (
const Record::Field &Field : llvm::reverse(R->
fields())) {
5609 if (!
D->isPrimitive() && !
D->isPrimitiveArray()) {
5620 for (
const Record::Base &
Base : llvm::reverse(R->
bases())) {
5621 if (
Base.R->isAnonymousUnion())
5626 if (!this->emitRecordDestruction(
Base.R, {}))
5633 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
5636template <
class Emitter>
5641 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
5642 return this->compileConstructor(Ctor);
5643 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
5644 return this->compileDestructor(Dtor);
5647 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F);
5649 return this->emitLambdaStaticInvokerBody(MD);
5652 if (
const auto *Body = F->
getBody())
5653 if (!visitStmt(Body))
5662template <
class Emitter>
5664 const Expr *SubExpr =
E->getSubExpr();
5666 return this->VisitComplexUnaryOperator(
E);
5668 return this->VisitVectorUnaryOperator(
E);
5670 return this->VisitFixedPointUnaryOperator(
E);
5671 std::optional<PrimType>
T = classify(SubExpr->
getType());
5673 switch (
E->getOpcode()) {
5675 if (!Ctx.getLangOpts().CPlusPlus14)
5676 return this->emitInvalid(
E);
5678 return this->emitError(
E);
5680 if (!this->visit(SubExpr))
5684 if (!this->emitIncPtr(
E))
5687 return DiscardResult ? this->emitPopPtr(
E) :
true;
5691 return DiscardResult ? this->emitIncfPop(getFPOptions(
E),
E)
5692 : this->emitIncf(getFPOptions(
E),
E);
5695 return DiscardResult ? this->emitIncPop(*
T,
E) : this->emitInc(*
T,
E);
5698 if (!Ctx.getLangOpts().CPlusPlus14)
5699 return this->emitInvalid(
E);
5701 return this->emitError(
E);
5703 if (!this->visit(SubExpr))
5707 if (!this->emitDecPtr(
E))
5710 return DiscardResult ? this->emitPopPtr(
E) :
true;
5714 return DiscardResult ? this->emitDecfPop(getFPOptions(
E),
E)
5715 : this->emitDecf(getFPOptions(
E),
E);
5718 return DiscardResult ? this->emitDecPop(*
T,
E) : this->emitDec(*
T,
E);
5721 if (!Ctx.getLangOpts().CPlusPlus14)
5722 return this->emitInvalid(
E);
5724 return this->emitError(
E);
5726 if (!this->visit(SubExpr))
5730 if (!this->emitLoadPtr(
E))
5732 if (!this->emitConstUint8(1,
E))
5734 if (!this->emitAddOffsetUint8(
E))
5736 return DiscardResult ? this->emitStorePopPtr(
E) : this->emitStorePtr(
E);
5740 if (DiscardResult) {
5742 return this->emitIncfPop(getFPOptions(
E),
E);
5743 return this->emitIncPop(*
T,
E);
5747 const auto &TargetSemantics = Ctx.getFloatSemantics(
E->
getType());
5748 if (!this->emitLoadFloat(
E))
5750 if (!this->emitConstFloat(llvm::APFloat(TargetSemantics, 1),
E))
5752 if (!this->emitAddf(getFPOptions(
E),
E))
5754 if (!this->emitStoreFloat(
E))
5758 if (!this->emitLoad(*
T,
E))
5760 if (!this->emitConst(1,
E))
5762 if (!this->emitAdd(*
T,
E))
5764 if (!this->emitStore(*
T,
E))
5770 if (!Ctx.getLangOpts().CPlusPlus14)
5771 return this->emitInvalid(
E);
5773 return this->emitError(
E);
5775 if (!this->visit(SubExpr))
5779 if (!this->emitLoadPtr(
E))
5781 if (!this->emitConstUint8(1,
E))
5783 if (!this->emitSubOffsetUint8(
E))
5785 return DiscardResult ? this->emitStorePopPtr(
E) : this->emitStorePtr(
E);
5789 if (DiscardResult) {
5791 return this->emitDecfPop(getFPOptions(
E),
E);
5792 return this->emitDecPop(*
T,
E);
5796 const auto &TargetSemantics = Ctx.getFloatSemantics(
E->
getType());
5797 if (!this->emitLoadFloat(
E))
5799 if (!this->emitConstFloat(llvm::APFloat(TargetSemantics, 1),
E))
5801 if (!this->emitSubf(getFPOptions(
E),
E))
5803 if (!this->emitStoreFloat(
E))
5807 if (!this->emitLoad(*
T,
E))
5809 if (!this->emitConst(1,
E))
5811 if (!this->emitSub(*
T,
E))
5813 if (!this->emitStore(*
T,
E))
5820 return this->emitError(
E);
5823 return this->discard(SubExpr);
5825 if (!this->visitBool(SubExpr))
5828 if (!this->emitInv(
E))
5832 return this->emitCast(
PT_Bool, ET,
E);
5836 return this->emitError(
E);
5838 if (!this->visit(SubExpr))
5840 return DiscardResult ? this->emitPop(*
T,
E) : this->emitNeg(*
T,
E);
5843 return this->emitError(
E);
5845 if (!this->visit(SubExpr))
5847 return DiscardResult ? this->emitPop(*
T,
E) :
true;
5852 return this->emitGetMemberPtr(cast<DeclRefExpr>(SubExpr)->getDecl(),
E);
5855 return this->delegate(SubExpr);
5857 if (DiscardResult) {
5859 return this->discard(SubExpr);
5862 if (!this->visit(SubExpr))
5864 if (classifyPrim(SubExpr) ==
PT_Ptr)
5865 return this->emitNarrowPtr(
E);
5870 return this->emitError(
E);
5872 if (!this->visit(SubExpr))
5874 return DiscardResult ? this->emitPop(*
T,
E) : this->emitComp(*
T,
E);
5877 return this->delegate(SubExpr);
5880 if (!this->discard(SubExpr))
5882 return this->visitZeroInitializer(*
T, SubExpr->
getType(), SubExpr);
5885 return this->delegate(SubExpr);
5887 assert(
false &&
"Unhandled opcode");
5893template <
class Emitter>
5895 const Expr *SubExpr =
E->getSubExpr();
5899 return this->discard(SubExpr);
5901 std::optional<PrimType> ResT = classify(
E);
5902 auto prepareResult = [=]() ->
bool {
5904 std::optional<unsigned> LocalIndex = allocateLocal(SubExpr);
5907 return this->emitGetPtrLocal(*LocalIndex,
E);
5914 unsigned SubExprOffset = ~0u;
5915 auto createTemp = [=, &SubExprOffset]() ->
bool {
5916 SubExprOffset = this->allocateLocalPrimitive(SubExpr,
PT_Ptr,
true,
false);
5917 if (!this->visit(SubExpr))
5919 return this->emitSetLocal(
PT_Ptr, SubExprOffset,
E);
5923 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
5924 if (!this->emitGetLocal(
PT_Ptr, Offset,
E))
5926 return this->emitArrayElemPop(ElemT, Index,
E);
5929 switch (
E->getOpcode()) {
5931 if (!prepareResult())
5935 for (
unsigned I = 0; I != 2; ++I) {
5936 if (!getElem(SubExprOffset, I))
5938 if (!this->emitNeg(ElemT,
E))
5940 if (!this->emitInitElem(ElemT, I,
E))
5948 return this->delegate(SubExpr);
5951 if (!this->visit(SubExpr))
5953 if (!this->emitComplexBoolCast(SubExpr))
5955 if (!this->emitInv(
E))
5958 return this->emitCast(
PT_Bool, ET,
E);
5962 return this->emitComplexReal(SubExpr);
5965 if (!this->visit(SubExpr))
5969 if (!this->emitConstUint8(1,
E))
5971 return this->emitArrayElemPtrPopUint8(
E);
5976 return this->emitArrayElemPop(classifyPrim(
E->
getType()), 1,
E);
5979 if (!this->visit(SubExpr))
5982 if (!this->emitArrayElem(ElemT, 1,
E))
5984 if (!this->emitNeg(ElemT,
E))
5986 if (!this->emitInitElem(ElemT, 1,
E))
5988 return DiscardResult ? this->emitPopPtr(
E) :
true;
5991 return this->delegate(SubExpr);
5994 return this->emitInvalid(
E);
6000template <
class Emitter>
6002 const Expr *SubExpr =
E->getSubExpr();
6006 return this->discard(SubExpr);
6008 auto UnaryOp =
E->getOpcode();
6009 if (UnaryOp == UO_Extension)
6010 return this->delegate(SubExpr);
6012 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
6013 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
6014 return this->emitInvalid(
E);
6017 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
6018 return this->delegate(SubExpr);
6021 std::optional<unsigned> LocalIndex = allocateLocal(SubExpr);
6024 if (!this->emitGetPtrLocal(*LocalIndex,
E))
6029 unsigned SubExprOffset =
6030 this->allocateLocalPrimitive(SubExpr,
PT_Ptr,
true,
false);
6031 if (!this->visit(SubExpr))
6033 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset,
E))
6038 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
6039 if (!this->emitGetLocal(
PT_Ptr, Offset,
E))
6041 return this->emitArrayElemPop(ElemT, Index,
E);
6046 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6047 if (!getElem(SubExprOffset, I))
6049 if (!this->emitNeg(ElemT,
E))
6051 if (!this->emitInitElem(ElemT, I,
E))
6066 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6067 if (!getElem(SubExprOffset, I))
6070 if (!this->emitPrimCast(ElemT,
PT_Bool, Ctx.getASTContext().BoolTy,
E))
6072 if (!this->emitInv(
E))
6074 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(),
E))
6076 if (!this->emitNeg(ElemT,
E))
6078 if (ElemT != ResultVecElemT &&
6079 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy,
E))
6081 if (!this->emitInitElem(ResultVecElemT, I,
E))
6087 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
6088 if (!getElem(SubExprOffset, I))
6091 if (!this->emitInv(
E))
6094 if (!this->emitComp(ElemT,
E))
6097 if (!this->emitInitElem(ElemT, I,
E))
6102 llvm_unreachable(
"Unsupported unary operators should be handled up front");
6107template <
class Emitter>
6112 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(
D)) {
6113 return this->emitConst(ECD->getInitVal(),
E);
6114 }
else if (
const auto *BD = dyn_cast<BindingDecl>(
D)) {
6115 return this->visit(BD->getBinding());
6116 }
else if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(
D)) {
6117 const Function *F = getFunction(FuncDecl);
6118 return F && this->emitGetFnPtr(F,
E);
6119 }
else if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(
D)) {
6120 if (std::optional<unsigned> Index =
P.getOrCreateGlobal(
D)) {
6121 if (!this->emitGetPtrGlobal(*Index,
E))
6123 if (std::optional<PrimType>
T = classify(
E->
getType())) {
6124 if (!this->visitAPValue(TPOD->getValue(), *
T,
E))
6126 return this->emitInitGlobal(*
T, *Index,
E);
6128 return this->visitAPValueInitializer(TPOD->getValue(),
E);
6137 bool IsReference =
D->getType()->isReferenceType();
6140 if (
auto It = Locals.find(
D); It != Locals.end()) {
6141 const unsigned Offset = It->second.Offset;
6143 return this->emitGetLocal(
PT_Ptr, Offset,
E);
6144 return this->emitGetPtrLocal(Offset,
E);
6145 }
else if (
auto GlobalIndex =
P.getGlobal(
D)) {
6147 if (!Ctx.getLangOpts().CPlusPlus11)
6148 return this->emitGetGlobal(classifyPrim(
E), *GlobalIndex,
E);
6149 return this->emitGetGlobalUnchecked(classifyPrim(
E), *GlobalIndex,
E);
6152 return this->emitGetPtrGlobal(*GlobalIndex,
E);
6153 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(
D)) {
6154 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
6155 if (IsReference || !It->second.IsPtr)
6156 return this->emitGetParam(classifyPrim(
E), It->second.Offset,
E);
6158 return this->emitGetPtrParam(It->second.Offset,
E);
6161 if (
D->getType()->isReferenceType())
6166 auto revisit = [&](
const VarDecl *VD) ->
bool {
6167 auto VarState = this->visitDecl(VD);
6169 if (VarState.notCreated())
6174 return this->visitDeclRef(
D,
E);
6178 if (
auto It = this->LambdaCaptures.find(
D);
6179 It != this->LambdaCaptures.end()) {
6180 auto [Offset, IsPtr] = It->second;
6183 return this->emitGetThisFieldPtr(Offset,
E);
6184 return this->emitGetPtrThisField(Offset,
E);
6185 }
else if (
const auto *DRE = dyn_cast<DeclRefExpr>(
E);
6186 DRE && DRE->refersToEnclosingVariableOrCapture()) {
6187 if (
const auto *VD = dyn_cast<VarDecl>(
D); VD && VD->isInitCapture())
6191 if (
D != InitializingDecl) {
6194 if (Ctx.getLangOpts().CPlusPlus) {
6195 if (
const auto *VD = dyn_cast<VarDecl>(
D)) {
6196 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
6197 if (
T.isConstant(Ctx.getASTContext()))
6203 if (isa<DecompositionDecl>(VD))
6206 if ((VD->hasGlobalStorage() || VD->isStaticDataMember()) &&
6207 typeShouldBeVisited(VD->getType()))
6214 if (VD->isLocalVarDecl() && typeShouldBeVisited(VD->getType()) &&
6215 VD->getInit() && !VD->getInit()->isValueDependent()) {
6217 if (VD->evaluateValue())
6220 if (!
D->getType()->isReferenceType())
6221 return this->emitDummyPtr(
D,
E);
6223 return this->emitInvalidDeclRef(cast<DeclRefExpr>(
E),
6228 if (
const auto *VD = dyn_cast<VarDecl>(
D);
6229 VD && VD->getAnyInitializer() &&
6230 VD->getType().isConstant(Ctx.getASTContext()) && !VD->isWeak())
6235 return this->emitDummyPtr(
D,
E);
6238template <
class Emitter>
6240 const auto *
D =
E->getDecl();
6241 return this->visitDeclRef(
D,
E);
6246 C->emitDestruction();
6249template <
class Emitter>
6253 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
6255 return Ty->getAsCXXRecordDecl();
6257 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
6258 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
6260 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
6264template <
class Emitter>
6271 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
6276 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
6277 getFPOptions(
E),
E);
6279 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
6280 getFPOptions(
E),
E);
6284 return this->emitCastFloatingIntegral(ToT, getFPOptions(
E),
E);
6289 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT),
E);
6291 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT),
E);
6295 return FromT != ToT ? this->emitCast(FromT, ToT,
E) :
true;
6299 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
6300 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(
E),
E);
6308template <
class Emitter>
6313 return this->discard(SubExpr);
6315 if (!this->visit(SubExpr))
6318 if (!this->emitConstUint8(0, SubExpr))
6320 return this->emitArrayElemPtrPopUint8(SubExpr);
6324 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
6328template <
class Emitter>
6330 assert(!DiscardResult);
6334 if (!this->emitArrayElem(ElemT, 0,
E))
6337 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(
E),
E))
6340 if (!this->emitCast(ElemT,
PT_Bool,
E))
6345 LabelTy LabelTrue = this->getLabel();
6346 if (!this->jumpTrue(LabelTrue))
6349 if (!this->emitArrayElemPop(ElemT, 1,
E))
6352 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(
E),
E))
6355 if (!this->emitCast(ElemT,
PT_Bool,
E))
6359 LabelTy EndLabel = this->getLabel();
6360 this->jump(EndLabel);
6362 this->emitLabel(LabelTrue);
6363 if (!this->emitPopPtr(
E))
6365 if (!this->emitConstBool(
true,
E))
6368 this->fallthrough(EndLabel);
6369 this->emitLabel(EndLabel);
6374template <
class Emitter>
6377 assert(
E->isComparisonOp());
6379 assert(!DiscardResult);
6385 LHSIsComplex =
true;
6386 ElemT = classifyComplexElementType(LHS->
getType());
6387 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true,
6389 if (!this->visit(LHS))
6391 if (!this->emitSetLocal(
PT_Ptr, LHSOffset,
E))
6394 LHSIsComplex =
false;
6396 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true,
false);
6397 if (!this->visit(LHS))
6399 if (!this->emitSetLocal(LHST, LHSOffset,
E))
6406 RHSIsComplex =
true;
6407 ElemT = classifyComplexElementType(RHS->
getType());
6408 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true,
6410 if (!this->visit(RHS))
6412 if (!this->emitSetLocal(
PT_Ptr, RHSOffset,
E))
6415 RHSIsComplex =
false;
6417 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true,
false);
6418 if (!this->visit(RHS))
6420 if (!this->emitSetLocal(RHST, RHSOffset,
E))
6424 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
6425 bool IsComplex) ->
bool {
6427 if (!this->emitGetLocal(
PT_Ptr, LocalOffset,
E))
6429 return this->emitArrayElemPop(ElemT, Index,
E);
6431 return this->emitGetLocal(ElemT, LocalOffset,
E);
6434 for (
unsigned I = 0; I != 2; ++I) {
6436 if (!getElem(LHSOffset, I, LHSIsComplex))
6438 if (!getElem(RHSOffset, I, RHSIsComplex))
6441 if (!this->emitEQ(ElemT,
E))
6444 if (!this->emitCastBoolUint8(
E))
6449 if (!this->emitAddUint8(
E))
6451 if (!this->emitConstUint8(2,
E))
6454 if (
E->getOpcode() == BO_EQ) {
6455 if (!this->emitEQUint8(
E))
6457 }
else if (
E->getOpcode() == BO_NE) {
6458 if (!this->emitNEUint8(
E))
6465 return this->emitCast(
PT_Bool, ResT,
E);
6472template <
class Emitter>
6481 const Function *DtorFunc = getFunction(Dtor);
6486 if (!this->emitDupPtr(
Loc))
6488 return this->emitCall(DtorFunc, 0,
Loc);
6493template <
class Emitter>
6518 for (ssize_t I = Desc->
getNumElems() - 1; I >= 0; --I) {
6519 if (!this->emitConstUint64(I,
Loc))
6521 if (!this->emitArrayElemPtrUint64(
Loc))
6523 if (!this->emitDestruction(ElemDesc,
Loc))
6525 if (!this->emitPopPtr(
Loc))
6540template <
class Emitter>
6542 assert(!DiscardResult &&
"Should've been checked before");
6544 unsigned DummyID =
P.getOrCreateDummy(
D);
6546 if (!this->emitGetPtrGlobal(DummyID,
E))
6554 return this->emitDecayPtr(
PT_Ptr, PT,
E);
6567template <
class Emitter>
6569 const Expr *SubExpr =
E->getSubExpr();
6572 std::optional<PrimType> ToT = classify(ToType);
6578 std::optional<unsigned> LocalIndex = allocateLocal(
E);
6581 if (!this->emitGetPtrLocal(*LocalIndex,
E))
6591 if (!this->visit(SubExpr))
6593 }
else if (std::optional<PrimType> FromT = classify(SubExpr)) {
6594 unsigned TempOffset = allocateLocalPrimitive(
6595 SubExpr, *FromT,
true,
false);
6596 if (!this->visit(SubExpr))
6598 if (!this->emitSetLocal(*FromT, TempOffset,
E))
6600 if (!this->emitGetPtrLocal(TempOffset,
E))
6607 if (!this->emitBitCast(
E))
6609 return DiscardResult ? this->emitPopPtr(
E) :
true;
6613 const llvm::fltSemantics *TargetSemantics =
nullptr;
6615 TargetSemantics = &Ctx.getFloatSemantics(ToType);
6621 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
6623 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
6624 ResultBitWidth, TargetSemantics,
E))
6628 return this->emitPop(*ToT,
E);
ASTImporterLookupTable & LT
#define EMIT_ARITH_OP(OP)
static CharUnits AlignOfType(QualType T, const ASTContext &ASTCtx, UnaryExprOrTypeTrait Kind)
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.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
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...
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.
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
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.
A builtin binary operation expression such as "x + y" or "x <= y".
static bool isLogicalOp(Opcode Opc)
static bool isComparisonOp(Opcode Opc)
static bool isCommaOp(Opcode Opc)
static Opcode getOpForCompoundAssignment(Opcode Opc)
static bool isPtrMemOp(Opcode Opc)
predicates to categorize the respective opcodes.
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
BreakStmt - This represents a break.
Represents a base class of a C++ class.
Represents binding an expression to a temporary.
A boolean literal, per ([C++ lex.bool] Boolean literals).
Represents a call to a C++ constructor.
Represents a C++ constructor within a class.
bool isCopyOrMoveConstructor(unsigned &TypeQuals) const
Determine whether this is a copy or move constructor.
A default argument (C++ [dcl.fct.default]).
A use of a default initializer in a constructor or in aggregate initialization.
Represents a delete expression for memory deallocation and destructor calls, e.g.
Represents a C++ destructor within a class.
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
Represents a call to an inherited base class constructor from an inheriting constructor.
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 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)".
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
The null pointer literal (C++11 [lex.nullptr])
Represents a list-initialization with parenthesis.
Represents a C++ struct/union/class.
capture_const_iterator captures_end() const
capture_const_iterator captures_begin() 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.
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]).
CXXTryStmt - A C++ try block, including all handlers.
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
A Microsoft C++ __uuidof expression, which gets the _GUID that corresponds to the supplied type or ex...
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
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.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Complex values, per C99 6.2.5p11.
QualType getElementType() const
CompoundAssignOperator - For compound assignments (e.g.
CompoundLiteralExpr - [C99 6.5.2.5].
CompoundStmt - This represents a group of statements like { stmt stmt }.
Stmt * getStmtExprResult()
Represents the specialization of a concept - evaluates to a prvalue of type bool.
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...
ContinueStmt - This represents a continue.
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
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...
DoStmt - This represents a 'do/while' stmt.
Represents a reference to #emded data.
bool isFixed() const
Returns true if this is an Objective-C, C++11, or Microsoft-style enumeration with a fixed underlying...
A helper class that allows the use of isa/cast/dyncast to detect TagType objects of enums.
EnumDecl * getDecl() const
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 * 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 ...
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.
bool isUnnamedBitField() const
Determines whether this is an unnamed bitfield.
ForStmt - This represents a 'for (init;cond;inc)' stmt.
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.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
bool isReplaceableGlobalAllocationFunction(std::optional< unsigned > *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions: void *operato...
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.
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
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...
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
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.
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
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.
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
A pointer to member type per C++ 8.3.3 - Pointers to members.
const Type * getClass() const
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.
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
ObjCEncodeExpr, used for @encode in Objective-C.
ObjCStringLiteral, used for Objective-C string literals i.e.
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Helper class for OffsetOfExpr.
@ Array
An index into an array.
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
ParenExpr - This represents a parenthesized expression, e.g.
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.
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
A (possibly-)qualified type.
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.
QualType getCanonicalType() const
bool isConstQualified() const
Determine whether this type is const-qualified.
Represents a struct/union/class.
A helper class that allows the use of isa/cast/dyncast to detect TagType objects of structs/unions/cl...
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...
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
Scope - A scope is a transient data structure that is used while parsing the program.
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
Represents an expression that computes the length of a parameter pack.
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
Represents a C++11 static_assert declaration.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
Stmt - This represents one statement.
StringLiteral - This represents a string literal expression, e.g.
static StringLiteral * Create(const ASTContext &Ctx, StringRef Str, StringLiteralKind Kind, bool Pascal, QualType Ty, const SourceLocation *Loc, unsigned NumConcatenated)
This is the "fully general" constructor that allows representation of strings formed from multiple co...
Represents a reference to a non-type template parameter that has been substituted with a template arg...
SwitchStmt - This represents a 'switch' stmt.
Represents the declaration of a struct/union/class/enum.
bool isCompleteDefinition() const
Return true if this decl has its body fully specified.
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
The base class of the type hierarchy.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
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
bool isVoidPointerType() const
bool isConstantSizeType() const
Return true if this is not a variable sized type, according to the rules of C99 6....
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
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 isFloatingType() const
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
Base class for declarations which introduce a typedef-name.
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Represents a C++ using-enum-declaration.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a variable declaration or definition.
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.
Scope for storage declared in a compound statement.
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.
OptLabelTy BreakLabel
Point to break to.
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 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)
bool visitDeclAndReturn(const VarDecl *VD, bool ConstantContext) override
Toplevel visitDeclAndReturn().
bool VisitTypeTraitExpr(const TypeTraitExpr *E)
bool VisitLambdaExpr(const LambdaExpr *E)
bool VisitMemberExpr(const MemberExpr *E)
bool VisitBinaryOperator(const BinaryOperator *E)
bool visitAttributedStmt(const AttributedStmt *S)
bool VisitPackIndexingExpr(const PackIndexingExpr *E)
bool VisitArraySubscriptExpr(const ArraySubscriptExpr *E)
bool VisitCallExpr(const CallExpr *E)
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 visitDeclStmt(const DeclStmt *DS)
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)
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)
OptLabelTy DefaultLabel
Default case label.
bool VisitStmtExpr(const StmtExpr *E)
std::optional< unsigned > allocateLocal(DeclTy &&Decl, QualType Ty=QualType(), const ValueDecl *ExtendingDecl=nullptr)
Allocates a space storing a local given its type.
bool VisitFixedPointLiteral(const FixedPointLiteral *E)
bool VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *E)
VariableScope< Emitter > * VarScope
Current scope.
VariableScope< Emitter > * ContinueVarScope
Scope to cleanup until when we see a continue statement.
bool VisitCXXNewExpr(const CXXNewExpr *E)
bool VisitCompoundAssignOperator(const CompoundAssignOperator *E)
bool visitArrayElemInit(unsigned ElemIndex, const Expr *Init)
Pointer to the array(not the element!) must be on the stack when calling this.
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.
bool visit(const Expr *E)
Evaluates an expression and places the result on the stack.
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 VisitExtVectorElementExpr(const ExtVectorElementExpr *E)
bool VisitObjCStringLiteral(const ObjCStringLiteral *E)
bool VisitArrayTypeTraitExpr(const ArrayTypeTraitExpr *E)
bool visitInitializer(const Expr *E)
Compiles an initializer.
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 VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *E)
bool visitDefaultStmt(const DefaultStmt *S)
typename Emitter::LabelTy LabelTy
VarCreationState visitDecl(const VarDecl *VD)
bool visitStmt(const Stmt *S)
bool VisitExpressionTraitExpr(const ExpressionTraitExpr *E)
bool visitAPValueInitializer(const APValue &Val, const Expr *E)
bool VisitVectorUnaryOperator(const UnaryOperator *E)
bool VisitCXXConstructExpr(const CXXConstructExpr *E)
bool VisitCXXNullPtrLiteralExpr(const CXXNullPtrLiteralExpr *E)
bool VisitObjCBoxedExpr(const ObjCBoxedExpr *E)
bool VisitCXXInheritedCtorInitExpr(const CXXInheritedCtorInitExpr *E)
bool VisitRecoveryExpr(const RecoveryExpr *E)
bool VisitRequiresExpr(const RequiresExpr *E)
bool Initializing
Flag inidicating if we're initializing an already created variable.
bool visitReturnStmt(const ReturnStmt *RS)
bool VisitCXXThrowExpr(const CXXThrowExpr *E)
bool VisitSubstNonTypeTemplateParmExpr(const SubstNonTypeTemplateParmExpr *E)
bool VisitChooseExpr(const ChooseExpr *E)
bool visitFunc(const FunctionDecl *F) override
bool visitCXXForRangeStmt(const CXXForRangeStmt *S)
bool visitCaseStmt(const CaseStmt *S)
bool VisitComplexBinOp(const BinaryOperator *E)
bool VisitAbstractConditionalOperator(const AbstractConditionalOperator *E)
bool VisitCXXTypeidExpr(const CXXTypeidExpr *E)
bool VisitBuiltinCallExpr(const CallExpr *E, unsigned BuiltinID)
OptLabelTy ContinueLabel
Point to continue to.
bool VisitImplicitValueInitExpr(const ImplicitValueInitExpr *E)
bool VisitCXXRewrittenBinaryOperator(const CXXRewrittenBinaryOperator *E)
bool VisitUnaryOperator(const UnaryOperator *E)
bool VisitFloatCompoundAssignOperator(const CompoundAssignOperator *E)
bool VisitGenericSelectionExpr(const GenericSelectionExpr *E)
bool visitDoStmt(const DoStmt *S)
bool VisitIntegerLiteral(const IntegerLiteral *E)
bool VisitInitListExpr(const InitListExpr *E)
bool VisitVectorBinOp(const BinaryOperator *E)
bool VisitStringLiteral(const StringLiteral *E)
bool VisitParenExpr(const ParenExpr *E)
bool VisitCXXNoexceptExpr(const CXXNoexceptExpr *E)
bool VisitShuffleVectorExpr(const ShuffleVectorExpr *E)
bool VisitPointerCompoundAssignOperator(const CompoundAssignOperator *E)
VariableScope< Emitter > * BreakVarScope
Scope to cleanup until when we see a break statement.
std::optional< LabelTy > OptLabelTy
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)
unsigned allocateTemporary(const Expr *E)
bool VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *E)
bool visitSwitchStmt(const SwitchStmt *S)
bool VisitCXXUuidofExpr(const CXXUuidofExpr *E)
unsigned allocateLocalPrimitive(DeclTy &&Decl, PrimType Ty, bool IsConst, bool IsExtended=false)
Creates a local primitive value.
bool VisitExprWithCleanups(const ExprWithCleanups *E)
bool visitWhileStmt(const WhileStmt *S)
bool visitIfStmt(const IfStmt *IS)
bool VisitAddrLabelExpr(const AddrLabelExpr *E)
bool VisitFloatingLiteral(const FloatingLiteral *E)
bool VisitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *E)
bool VisitGNUNullExpr(const GNUNullExpr *E)
bool VisitImaginaryLiteral(const ImaginaryLiteral *E)
bool VisitSYCLUniqueStableNameExpr(const SYCLUniqueStableNameExpr *E)
VarCreationState visitVarDecl(const VarDecl *VD, bool Toplevel=false)
Creates and initializes a variable from the given decl.
bool visitCXXTryStmt(const CXXTryStmt *S)
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.
void addExtended(const Scope::Local &Local) override
DeclScope(Compiler< Emitter > *Ctx, const ValueDecl *VD)
Wrapper around fixed point types.
static FixedPoint zero(llvm::FixedPointSemantics Sem)
unsigned getNumParams() const
bool hasThisPointer() const
bool hasRVO() const
Checks if the first argument is a RVO pointer.
Scope managing label targets.
LabelScope(Compiler< Emitter > *Ctx)
Compiler< Emitter > * Ctx
Compiler instance.
Generic scope for local variables.
bool destroyLocals(const Expr *E=nullptr) override
Explicit destruction of local variables.
void addLocal(const Scope::Local &Local) override
Sets the context for break/continue statements.
typename Compiler< Emitter >::LabelTy LabelTy
LoopScope(Compiler< Emitter > *Ctx, LabelTy BreakLabel, LabelTy ContinueLabel)
typename Compiler< Emitter >::OptLabelTy OptLabelTy
Scope used to handle initialization methods.
OptionScope(Compiler< Emitter > *Ctx, bool NewDiscardResult, bool NewInitializing)
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.
llvm::iterator_range< const_base_iter > bases() const
const Base * getVirtualBase(const RecordDecl *RD) const
Returns a virtual base descriptor.
unsigned getNumFields() const
bool isAnonymousUnion() const
Checks if the record is an anonymous union.
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
SwitchScope(Compiler< Emitter > *Ctx, CaseMap &&CaseLabels, LabelTy BreakLabel, OptLabelTy DefaultLabel)
typename Compiler< Emitter >::CaseMap CaseMap
Scope chain managing the variable lifetimes.
Compiler< Emitter > * Ctx
Compiler instance.
static llvm::RoundingMode getRoundingMode(FPOptions FPO)
bool Div(InterpState &S, CodePtr OpPC)
1) Pops the RHS from the stack.
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)
bool LE(InterpState &S, CodePtr OpPC)
PrimType
Enumeration of the primitive types of the VM.
bool Mul(InterpState &S, CodePtr OpPC)
llvm::BitVector collectNonNullArgs(const FunctionDecl *F, const llvm::ArrayRef< const Expr * > &Args)
size_t primSize(PrimType Type)
Returns the size of a primitive type in bytes.
llvm::PointerUnion< const Decl *, const Expr * > DeclTy
constexpr bool isIntegralType(PrimType T)
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
UnaryExprOrTypeTrait
Names for the "expression or type" traits.
@ SD_Static
Static storage duration.
const FunctionProtoType * T
@ Success
Template argument deduction was successful.
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.
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.
const Record *const ElemRecord
Pointer to the record, if block contains records.
bool isArray() const
Checks if the descriptor is of an array.
Descriptor used for global variables.
GlobalInitState InitState
static InitLink InitList()
static InitLink Elem(unsigned Index)
bool emit(Compiler< Emitter > *Ctx, const Expr *E) const
static InitLink Field(unsigned Offset)
static InitLink Decl(const ValueDecl *D)
static InitLink Temp(unsigned Offset)
Information about a local's storage.
State encapsulating if a the variable creation has been successful, unsuccessful, or no variable has ...
static VarCreationState NotCreated()