19#include "llvm/Support/SaveAndRestore.h"
30 if (
const auto *CE = dyn_cast_if_present<ConstantExpr>(E);
31 CE && CE->hasAPValueResult() &&
33 return CE->getResultAsAPSInt().getBoolValue();
44 OldInitializingDecl(
Ctx->InitializingDecl) {
45 Ctx->InitializingDecl = VD;
50 this->
Ctx->InitializingDecl = OldInitializingDecl;
51 this->
Ctx->InitStack.pop_back();
64 bool NewInitializing,
bool NewToLValue)
65 : Ctx(Ctx), OldDiscardResult(Ctx->DiscardResult),
66 OldInitializing(Ctx->
Initializing), OldToLValue(Ctx->ToLValue) {
67 Ctx->DiscardResult = NewDiscardResult;
68 Ctx->Initializing = NewInitializing;
69 Ctx->ToLValue = NewToLValue;
73 Ctx->DiscardResult = OldDiscardResult;
74 Ctx->Initializing = OldInitializing;
75 Ctx->ToLValue = OldToLValue;
82 bool OldDiscardResult;
87template <
class Emitter>
91 return Ctx->emitThis(E);
94 return Ctx->emitGetPtrFieldPop(
Offset, E);
96 return Ctx->emitGetPtrLocal(
Offset, E);
100 if (!Ctx->emitConstUint32(
Offset, E))
102 return Ctx->emitArrayElemPtrPopUint32(E);
104 return Ctx->emitRVOPtr(E);
108 llvm_unreachable(
"Unhandled InitLink kind");
124 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
125 assert(LI.Name != Name);
128 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel, ContinueLabel,
149 : Ctx(Ctx), OldCaseLabels(
std::move(this->Ctx->CaseLabels)) {
151 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
152 assert(LI.Name != Name);
155 this->Ctx->CaseLabels = std::move(CaseLabels);
156 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel,
158 DefaultLabel, Ctx->VarScope);
162 this->Ctx->CaseLabels = std::move(OldCaseLabels);
163 this->Ctx->LabelInfoStack.pop_back();
168 CaseMap OldCaseLabels;
174 Ctx->InStmtExpr =
true;
192 : Ctx(Ctx), OldFlag(Ctx->LocOverride), Enabled(Enabled) {
195 Ctx->LocOverride = NewValue;
200 Ctx->LocOverride = OldFlag;
205 std::optional<SourceInfo> OldFlag;
212template <
class Emitter>
220 case CK_LValueToRValue: {
231 if (
const auto *DRE = dyn_cast<DeclRefExpr>(SubExpr)) {
237 if (
auto GlobalIndex =
P.getGlobal(D))
238 return this->emitGetGlobal(*SubExprT, *GlobalIndex, E);
239 }
else if (
auto It =
Locals.find(D); It !=
Locals.end()) {
240 return this->emitGetLocal(*SubExprT, It->second.Offset, E);
241 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
242 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
243 return this->emitGetParam(*SubExprT, It->second.Index, E);
255 if (!this->emitGetPtrLocal(*LocalIndex, E))
259 if (!this->
visit(SubExpr))
263 return this->emitLoadPop(*SubExprT, E);
268 return this->emitMemcpy(E);
271 case CK_DerivedToBaseMemberPointer: {
282 ->getMostRecentCXXRecordDecl();
284 const CXXRecordDecl *ToDecl = B->getType()->getAsCXXRecordDecl();
285 unsigned DerivedOffset =
Ctx.collectBaseOffset(ToDecl, CurDecl);
287 if (!this->emitCastMemberPtrBasePop(DerivedOffset, ToDecl, E))
295 case CK_BaseToDerivedMemberPointer: {
306 ->getMostRecentCXXRecordDecl();
310 typedef std::reverse_iterator<CastExpr::path_const_iterator> ReverseIter;
312 PathI != PathE; ++PathI) {
313 const CXXRecordDecl *ToDecl = (*PathI)->getType()->getAsCXXRecordDecl();
314 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
316 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
323 assert(ToDecl != CurDecl);
324 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
326 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
332 case CK_UncheckedDerivedToBase:
333 case CK_DerivedToBase: {
338 if (
const auto *PT = dyn_cast<PointerType>(Ty))
339 return PT->getPointeeType()->getAsCXXRecordDecl();
340 return Ty->getAsCXXRecordDecl();
347 if (B->isVirtual()) {
348 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), E))
350 CurType = B->getType();
352 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
353 if (!this->emitGetPtrBasePop(
356 CurType = B->getType();
363 case CK_BaseToDerived: {
366 unsigned DerivedOffset =
372 return this->emitGetPtrDerivedPop(DerivedOffset,
377 case CK_FloatingCast: {
380 return this->emitVectorConversion(E->
getSubExpr(), E);
384 if (!this->
visit(SubExpr))
386 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
390 case CK_IntegralToFloating: {
392 return this->emitVectorConversion(E->
getSubExpr(), E);
395 if (!this->
visit(SubExpr))
397 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
398 return this->emitCastIntegralFloating(
classifyPrim(SubExpr),
399 TargetSemantics, getFPOptions(E), E);
402 case CK_FloatingToBoolean: {
404 return this->emitVectorConversion(E->
getSubExpr(), E);
408 if (
const auto *FL = dyn_cast<FloatingLiteral>(SubExpr))
409 return this->emitConstBool(FL->getValue().isNonZero(), E);
410 if (!this->
visit(SubExpr))
412 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
415 case CK_FloatingToIntegral: {
417 return this->emitVectorConversion(E->
getSubExpr(), E);
420 if (!this->
visit(SubExpr))
424 return this->emitCastFloatingIntegralAP(
Ctx.getBitWidth(E->
getType()),
427 return this->emitCastFloatingIntegralAPS(
Ctx.getBitWidth(E->
getType()),
430 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
433 case CK_NullToPointer:
434 case CK_NullToMemberPointer: {
437 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
442 case CK_PointerToIntegral: {
443 if (!this->
visit(SubExpr))
449 if (!this->emitDecayPtr(FromT,
PT_Ptr, E))
455 return this->emitCastPointerIntegralAP(
Ctx.getBitWidth(E->
getType()), E);
457 return this->emitCastPointerIntegralAPS(
Ctx.getBitWidth(E->
getType()), E);
458 return this->emitCastPointerIntegral(T, E);
461 case CK_ArrayToPointerDecay: {
462 if (!this->
visit(SubExpr))
464 return this->emitArrayDecay(E);
467 case CK_IntegralToPointer: {
469 assert(IntType->isIntegralOrEnumerationType());
470 if (!this->
visit(SubExpr))
484 return this->emitDecayPtr(
PT_Ptr, DestPtrT, E);
487 case CK_AtomicToNonAtomic:
488 case CK_ConstructorConversion:
489 case CK_FunctionToPointerDecay:
490 case CK_NonAtomicToAtomic:
492 case CK_UserDefinedConversion:
493 case CK_AddressSpaceConversion:
494 case CK_CPointerToObjCPointerCast:
511 return this->emitBuiltinBitCast(E);
526 if (!this->
visit(SubExpr))
534 return this->emitFnPtrCast(E);
541 if (!this->
visit(SubExpr))
543 return this->emitDecayPtr(*FromT, *ToT, E);
545 case CK_IntegralToBoolean:
546 case CK_FixedPointToBoolean: {
548 return this->emitVectorConversion(E->
getSubExpr(), E);
554 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr))
555 return this->emitConst(IL->getValue(), E);
556 if (!this->
visit(SubExpr))
561 case CK_IntegralCast:
563 return this->emitVectorConversion(E->
getSubExpr(), E);
565 case CK_BooleanToSignedIntegral: {
572 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr)) {
577 if (!this->emitConst(IL->getValue(), SubExpr))
580 if (!this->
visit(SubExpr))
588 if (!ED->isFixed()) {
589 if (!this->emitCheckEnumValue(*FromT, ED, E))
595 if (!this->emitCastAP(*FromT,
Ctx.getBitWidth(E->
getType()), E))
598 if (!this->emitCastAPS(*FromT,
Ctx.getBitWidth(E->
getType()), E))
603 if (!this->emitCast(*FromT, *ToT, E))
606 if (E->
getCastKind() == CK_BooleanToSignedIntegral)
607 return this->emitNeg(*ToT, E);
611 case CK_PointerToBoolean:
612 case CK_MemberPointerToBoolean: {
615 if (!this->
visit(SubExpr))
617 return this->emitIsNonNull(PtrT, E);
620 case CK_IntegralComplexToBoolean:
621 case CK_FloatingComplexToBoolean: {
622 if (!this->
visit(SubExpr))
624 return this->emitComplexBoolCast(SubExpr);
627 case CK_IntegralComplexToReal:
628 case CK_FloatingComplexToReal:
629 return this->emitComplexReal(SubExpr);
631 case CK_IntegralRealToComplex:
632 case CK_FloatingRealToComplex: {
639 if (!this->emitGetPtrLocal(*LocalIndex, E))
648 if (!this->visitZeroInitializer(T, SubExpr->
getType(), SubExpr))
650 return this->emitInitElem(T, 1, SubExpr);
653 case CK_IntegralComplexCast:
654 case CK_FloatingComplexCast:
655 case CK_IntegralComplexToFloatingComplex:
656 case CK_FloatingComplexToIntegralComplex: {
663 if (!this->emitGetPtrLocal(*LocalIndex, E))
670 unsigned SubExprOffset =
672 if (!this->
visit(SubExpr))
674 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
682 for (
unsigned I = 0; I != 2; ++I) {
683 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, E))
685 if (!this->emitArrayElemPop(SourceElemT, I, E))
689 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, E))
693 if (!this->emitInitElem(DestElemT, I, E))
699 case CK_VectorSplat: {
710 if (!this->emitGetPtrLocal(*LocalIndex, E))
716 unsigned ElemOffset =
720 if (!this->
visit(SubExpr))
725 if (!this->emitSetLocal(ElemT, ElemOffset, E))
728 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
729 if (!this->emitGetLocal(ElemT, ElemOffset, E))
731 if (!this->emitInitElem(ElemT, I, E))
738 case CK_HLSLVectorTruncation: {
743 if (!this->
visit(SubExpr))
745 return this->emitArrayElemPop(*ResultT, 0, E);
754 if (!this->emitGetPtrLocal(*LocalIndex, E))
759 if (!this->
visit(SubExpr))
761 return this->emitCopyArray(classifyVectorElementType(E->
getType()), 0, 0,
765 case CK_IntegralToFixedPoint: {
766 if (!this->
visit(SubExpr))
770 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
775 return this->emitPopFixedPoint(E);
778 case CK_FloatingToFixedPoint: {
779 if (!this->
visit(SubExpr))
783 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
784 if (!this->emitCastFloatingFixedPoint(Sem, E))
787 return this->emitPopFixedPoint(E);
790 case CK_FixedPointToFloating: {
791 if (!this->
visit(SubExpr))
793 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
794 if (!this->emitCastFixedPointFloating(TargetSemantics, E))
797 return this->emitPopFloat(E);
800 case CK_FixedPointToIntegral: {
801 if (!this->
visit(SubExpr))
804 if (!this->emitCastFixedPointIntegral(IntegralT, E))
807 return this->emitPop(IntegralT, E);
810 case CK_FixedPointCast: {
811 if (!this->
visit(SubExpr))
814 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
815 if (!this->emitCastFixedPoint(Sem, E))
818 return this->emitPopFixedPoint(E);
830 case CK_LValueBitCast:
835 case CK_HLSLArrayRValue: {
842 if (!this->emitGetPtrLocal(*LocalIndex, E))
845 if (!this->
visit(SubExpr))
847 return this->emitMemcpy(E);
850 case CK_HLSLMatrixTruncation: {
855 if (!this->
visit(SubExpr))
857 return this->emitArrayElemPop(*ResultT, 0, E);
866 if (!this->emitGetPtrLocal(*LocalIndex, E))
871 if (!this->
visit(SubExpr))
873 return this->emitCopyArray(classifyMatrixElementType(SubExpr->
getType()), 0,
877 case CK_HLSLAggregateSplatCast: {
887 if (!this->emitGetPtrLocal(*LocalIndex, E))
897 if (!this->
visit(SubExpr))
899 if (!this->emitSetLocal(SrcElemT, SrcOffset, E))
903 return emitHLSLAggregateSplat(SrcElemT, SrcOffset, E->
getType(), E);
906 case CK_HLSLElementwiseCast: {
917 unsigned SrcPtrOffset =
919 if (!this->
visit(SubExpr))
921 if (!this->emitSetLocal(
PT_Ptr, SrcPtrOffset, E))
925 if (!emitHLSLFlattenAggregate(SrcType, SrcPtrOffset, Elements, 1, E))
927 if (Elements.empty())
930 const HLSLFlatElement &Src = Elements[0];
931 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
933 return this->emitPrimCast(Src.Type, *DestT, DestType, E);
940 if (!this->emitGetPtrLocal(*LocalIndex, E))
946 if (!this->
visit(SubExpr))
948 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
952 unsigned ElemCount = countHLSLFlatElements(DestType);
955 Elements.reserve(ElemCount);
956 if (!emitHLSLFlattenAggregate(SrcType, SrcOffset, Elements, ElemCount, E))
961 assert(Elements.size() == ElemCount &&
962 "Source type has fewer scalar elements than the destination type");
964 return emitHLSLConstructAggregate(DestType, Elements, E);
971 const Record::Field *RF = R->getField(UnionField);
972 QualType FieldType = RF->Decl->getType();
975 if (!this->
visit(SubExpr))
977 if (RF->isBitField())
978 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
980 return this->emitInitFieldActivate(*PT, RF->Offset, E);
983 if (!this->emitGetPtrField(RF->Offset, E))
985 if (!this->emitActivate(E))
991 return this->emitInvalid(E);
993 llvm_unreachable(
"Unhandled clang::CastKind enum");
996template <
class Emitter>
998 return this->emitBuiltinBitCast(E);
1001template <
class Emitter>
1006 return this->emitConst(
LE->getValue(),
LE);
1009template <
class Emitter>
1015 return this->emitFloat(F, E);
1018template <
class Emitter>
1028 if (!this->emitGetPtrLocal(*LocalIndex, E))
1035 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
1037 if (!this->emitInitElem(SubExprT, 0, SubExpr))
1042template <
class Emitter>
1050 auto Sem =
Ctx.getASTContext().getFixedPointSemantics(E->
getType());
1055template <
class Emitter>
1060template <
class Emitter>
1087 return this->emitComplexComparison(LHS, RHS, E);
1095 if (!this->
visit(LHS))
1098 if (!this->
visit(RHS))
1101 if (!this->emitToMemberPtr(E))
1107 if (!this->emitCastMemberPtrPtr(E))
1124 Ctx.getASTContext().CompCategories.lookupInfoForType(E->
getType());
1130 if (!this->emitGetPtrLocal(*ResultIndex, E))
1137 return this->emitCMP3(*
LT, CmpInfo, E);
1140 if (!
LT || !RT || !T)
1150 return this->visitAssignment(LHS, RHS, E);
1157 auto MaybeCastToBool = [
this, T, E](
bool Result) {
1161 return this->emitPopBool(E);
1163 return this->emitCast(
PT_Bool, *T, E);
1167 auto Discard = [
this, T, E](
bool Result) {
1175 return MaybeCastToBool(this->emitEQ(*
LT, E));
1177 return MaybeCastToBool(this->emitNE(*
LT, E));
1179 return MaybeCastToBool(this->emitLT(*
LT, E));
1181 return MaybeCastToBool(this->emitLE(*
LT, E));
1183 return MaybeCastToBool(this->emitGT(*
LT, E));
1185 return MaybeCastToBool(this->emitGE(*
LT, E));
1188 return Discard(this->emitSubf(getFPOptions(E), E));
1189 return Discard(this->emitSub(*T, E));
1192 return Discard(this->emitAddf(getFPOptions(E), E));
1193 return Discard(this->emitAdd(*T, E));
1196 return Discard(this->emitMulf(getFPOptions(E), E));
1197 return Discard(this->emitMul(*T, E));
1199 return Discard(this->emitRem(*T, E));
1202 return Discard(this->emitDivf(getFPOptions(E), E));
1203 return Discard(this->emitDiv(*T, E));
1205 return Discard(this->emitBitAnd(*T, E));
1207 return Discard(this->emitBitOr(*T, E));
1209 return Discard(this->emitShl(*
LT, *RT, E));
1211 return Discard(this->emitShr(*
LT, *RT, E));
1213 return Discard(this->emitBitXor(*T, E));
1216 llvm_unreachable(
"Already handled earlier");
1221 llvm_unreachable(
"Unhandled binary op");
1226template <
class Emitter>
1232 if ((Op != BO_Add && Op != BO_Sub) ||
1243 auto visitAsPointer = [&](
const Expr *E,
PrimType T) ->
bool {
1244 if (!this->
visit(E))
1247 return this->emitDecayPtr(T,
PT_Ptr, E);
1256 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1264 ElemTypeSize =
Ctx.getASTContext().getTypeSizeInChars(ElemType);
1267 if (!this->emitSubPtr(IntT, ElemTypeSize.
isZero(), E))
1274 if (!visitAsPointer(RHS, *RT))
1276 if (!this->
visit(LHS))
1280 if (!visitAsPointer(LHS, *
LT))
1282 if (!this->
visit(RHS))
1293 if (!this->emitAddOffset(OffsetType, E))
1297 if (!this->emitSubOffset(OffsetType, E))
1314template <
class Emitter>
1324 LabelTy LabelTrue = this->getLabel();
1325 LabelTy LabelEnd = this->getLabel();
1329 if (!this->jumpTrue(LabelTrue, E))
1334 if (!this->jump(LabelEnd, E))
1337 this->emitLabel(LabelTrue);
1338 this->emitConstBool(
true, E);
1339 this->fallthrough(LabelEnd);
1340 this->emitLabel(LabelEnd);
1343 assert(Op == BO_LAnd);
1346 LabelTy LabelFalse = this->getLabel();
1347 LabelTy LabelEnd = this->getLabel();
1351 if (!this->jumpFalse(LabelFalse, E))
1356 if (!this->jump(LabelEnd, E))
1359 this->emitLabel(LabelFalse);
1360 this->emitConstBool(
false, E);
1361 this->fallthrough(LabelEnd);
1362 this->emitLabel(LabelEnd);
1366 return this->emitPopBool(E);
1371 return this->emitCast(
PT_Bool, *T, E);
1375template <
class Emitter>
1382 if (!this->emitGetPtrLocal(*LocalIndex, E))
1391 PrimType ResultElemT = this->classifyComplexElementType(E->
getType());
1392 unsigned ResultOffset = ~0u;
1398 if (!this->emitDupPtr(E))
1400 if (!this->emitSetLocal(
PT_Ptr, ResultOffset, E))
1405 LHSType = AT->getValueType();
1408 RHSType = AT->getValueType();
1417 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1422 if (!this->
visit(LHS))
1424 if (!this->
visit(RHS))
1426 if (!this->emitMulc(ElemT, E))
1429 return this->emitPopPtr(E);
1433 if (Op == BO_Div && RHSIsComplex) {
1440 if (!LHSIsComplex) {
1445 LHSOffset = *LocalIndex;
1447 if (!this->emitGetPtrLocal(LHSOffset, E))
1450 if (!this->
visit(LHS))
1453 if (!this->emitInitElem(ElemT, 0, E))
1456 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1458 if (!this->emitInitElem(ElemT, 1, E))
1461 if (!this->
visit(LHS))
1465 if (!this->
visit(RHS))
1467 if (!this->emitDivc(ElemT, E))
1470 return this->emitPopPtr(E);
1477 if (!this->
visit(LHS))
1479 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1484 if (!this->
visit(LHS))
1486 if (!this->emitSetLocal(LHST, LHSOffset, E))
1494 if (!this->
visit(RHS))
1496 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1501 if (!this->
visit(RHS))
1503 if (!this->emitSetLocal(RHST, RHSOffset, E))
1510 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1511 unsigned ElemIndex,
unsigned Offset,
1512 const Expr *E) ->
bool {
1514 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1516 return this->emitArrayElemPop(classifyComplexElementType(E->
getType()),
1519 if (ElemIndex == 0 || !LoadZero)
1526 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1529 if (!this->emitGetLocal(
PT_Ptr, ResultOffset, E))
1536 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1539 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1542 if (!this->emitAddf(getFPOptions(E), E))
1545 if (!this->emitAdd(ResultElemT, E))
1550 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1553 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1556 if (!this->emitSubf(getFPOptions(E), E))
1559 if (!this->emitSub(ResultElemT, E))
1564 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1567 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1571 if (!this->emitMulf(getFPOptions(E), E))
1574 if (!this->emitMul(ResultElemT, E))
1579 assert(!RHSIsComplex);
1580 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1583 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1587 if (!this->emitDivf(getFPOptions(E), E))
1590 if (!this->emitDiv(ResultElemT, E))
1601 if (!this->emitInitElemPop(ResultElemT, ElemIndex, E))
1604 if (!this->emitPop(ResultElemT, E))
1609 return this->emitPopPtr(E);
1615template <
class Emitter>
1620 "Comma op should be handled in VisitBinaryOperator");
1634 if (!this->emitGetPtrLocal(*LocalIndex, E))
1648 assert(
Ctx.getASTContext().hasSameUnqualifiedType(
1651 if (!this->
visit(LHS))
1653 if (!this->
visit(RHS))
1655 if (!this->emitCopyArray(ElemT, 0, 0, VecTy->getNumElements(), E))
1658 return this->emitPopPtr(E);
1663 unsigned LHSOffset =
1665 if (!this->
visit(LHS))
1667 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1671 unsigned RHSOffset =
1673 if (!this->
visit(RHS))
1675 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1687 if (NeedIntPromot) {
1689 Ctx.getASTContext().getPromotedIntegerType(
Ctx.getASTContext().BoolTy);
1694 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1695 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1697 if (!this->emitArrayElemPop(ElemT, Index, E))
1700 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
1702 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1704 }
else if (NeedIntPromot) {
1705 if (!this->emitPrimCast(ElemT, PromotT, PromotTy, E))
1711#define EMIT_ARITH_OP(OP) \
1713 if (ElemT == PT_Float) { \
1714 if (!this->emit##OP##f(getFPOptions(E), E)) \
1717 if (!this->emit##OP(ElemT, E)) \
1723 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1724 if (!getElem(LHSOffset, ElemT, I))
1726 if (!getElem(RHSOffset, RHSElemT, I))
1738 if (!this->emitRem(ElemT, E))
1742 if (!this->emitBitAnd(OpT, E))
1746 if (!this->emitBitOr(OpT, E))
1750 if (!this->emitBitXor(OpT, E))
1754 if (!this->emitShl(OpT, RHSElemT, E))
1758 if (!this->emitShr(OpT, RHSElemT, E))
1762 if (!this->emitEQ(ElemT, E))
1766 if (!this->emitNE(ElemT, E))
1770 if (!this->emitLE(ElemT, E))
1774 if (!this->emitLT(ElemT, E))
1778 if (!this->emitGE(ElemT, E))
1782 if (!this->emitGT(ElemT, E))
1787 if (!this->emitBitAnd(ResultElemT, E))
1792 if (!this->emitBitOr(ResultElemT, E))
1796 return this->emitInvalid(E);
1805 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1807 if (!this->emitNeg(ResultElemT, E))
1813 if (NeedIntPromot &&
1814 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(), E))
1818 if (!this->emitInitElem(ResultElemT, I, E))
1827template <
class Emitter>
1837 auto LHSSemaInt = LHSSema.toOpaqueInt();
1839 auto RHSSemaInt = RHSSema.toOpaqueInt();
1841 if (!this->
visit(LHS))
1849 if (!this->
visit(RHS))
1858 auto ConvertResult = [&](
bool R) ->
bool {
1862 auto CommonSema = LHSSema.getCommonSemantics(RHSSema).toOpaqueInt();
1863 if (ResultSema != CommonSema)
1864 return this->emitCastFixedPoint(ResultSema, E);
1868 auto MaybeCastToBool = [&](
bool Result) {
1873 return this->emitPop(T, E);
1875 return this->emitCast(
PT_Bool, T, E);
1881 return MaybeCastToBool(this->emitEQFixedPoint(E));
1883 return MaybeCastToBool(this->emitNEFixedPoint(E));
1885 return MaybeCastToBool(this->emitLTFixedPoint(E));
1887 return MaybeCastToBool(this->emitLEFixedPoint(E));
1889 return MaybeCastToBool(this->emitGTFixedPoint(E));
1891 return MaybeCastToBool(this->emitGEFixedPoint(E));
1893 return ConvertResult(this->emitAddFixedPoint(E));
1895 return ConvertResult(this->emitSubFixedPoint(E));
1897 return ConvertResult(this->emitMulFixedPoint(E));
1899 return ConvertResult(this->emitDivFixedPoint(E));
1901 return ConvertResult(this->emitShiftFixedPoint(
true, E));
1903 return ConvertResult(this->emitShiftFixedPoint(
false, E));
1906 return this->emitInvalid(E);
1909 llvm_unreachable(
"unhandled binop opcode");
1912template <
class Emitter>
1921 if (!this->
visit(SubExpr))
1923 if (!this->emitNegFixedPoint(E))
1926 return this->emitPopFixedPoint(E);
1932 llvm_unreachable(
"Unhandled unary opcode");
1935template <
class Emitter>
1944 return this->visitZeroInitializer(*T, QT, E);
1952 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1953 CXXRD && CXXRD->getNumVBases() > 0) {
1963 return this->visitZeroRecordInitializer(R, E);
1970 return this->visitZeroArrayInitializer(QT, E);
1974 QualType ElemQT = ComplexTy->getElementType();
1976 for (
unsigned I = 0; I < 2; ++I) {
1977 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1979 if (!this->emitInitElem(ElemT, I, E))
1986 unsigned NumVecElements = VecT->getNumElements();
1987 QualType ElemQT = VecT->getElementType();
1990 for (
unsigned I = 0; I < NumVecElements; ++I) {
1991 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1993 if (!this->emitInitElem(ElemT, I, E))
2000 unsigned NumElems = MT->getNumElementsFlattened();
2001 QualType ElemQT = MT->getElementType();
2004 for (
unsigned I = 0; I != NumElems; ++I) {
2005 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2007 if (!this->emitInitElem(ElemT, I, E))
2016template <
class Emitter>
2029 for (
const Expr *SubExpr : {LHS, RHS}) {
2030 if (!this->
visit(SubExpr)) {
2037 if (SubExpr ==
Base &&
Base->getType()->isPointerType()) {
2038 if (!this->emitExpandPtr(E))
2049 return this->emitError(E);
2052 if (!this->emitFlip(
PT_Ptr, *IndexT, E))
2056 if (!this->emitArrayElemPtrPop(*IndexT, E))
2059 return this->emitPopPtr(E);
2065 return this->emitLoadPop(*T, E);
2068template <
class Emitter>
2070 const Expr *ArrayFiller,
const Expr *E) {
2075 QT = AT->getValueType();
2078 if (
Inits.size() == 0)
2080 return this->emitInvalid(E);
2095 if (
Inits.size() == 0)
2096 return this->visitZeroInitializer(*T, QT, E);
2097 assert(
Inits.size() == 1);
2110 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
2117 bool BitField = FieldToInit->isBitField();
2119 return this->emitInitBitFieldActivate(T, FieldToInit->Offset,
2120 FieldToInit->bitWidth(), E);
2122 return this->emitInitBitField(T, FieldToInit->Offset,
2123 FieldToInit->bitWidth(), E);
2125 return this->emitInitFieldActivate(T, FieldToInit->Offset, E);
2126 return this->emitInitField(T, FieldToInit->Offset, E);
2129 auto initCompositeField = [=](
const Record::Field *FieldToInit,
2137 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
2140 if (
Activate && !this->emitActivate(E))
2147 if (
Inits.size() == 0) {
2148 if (!this->visitZeroRecordInitializer(R, E))
2153 if (
const auto *ILE = dyn_cast<InitListExpr>(E))
2154 FToInit = ILE->getInitializedFieldInUnion();
2158 const Record::Field *FieldToInit = R->getField(FToInit);
2160 if (!initPrimitiveField(FieldToInit,
Init, *T,
true))
2163 if (!initCompositeField(FieldToInit,
Init,
true))
2167 return this->emitFinishInit(E);
2170 assert(!R->isUnion());
2171 unsigned InitIndex = 0;
2174 while (InitIndex < R->getNumFields() &&
2175 R->getField(InitIndex)->isUnnamedBitField())
2186 const Record::Field *FieldToInit = R->getField(InitIndex);
2187 if (!initPrimitiveField(FieldToInit,
Init, *T))
2192 if (
const Record::Base *B = R->getBase(
Init->getType())) {
2193 if (!this->emitGetPtrBase(B->Offset,
Init))
2201 const Record::Field *FieldToInit = R->getField(InitIndex);
2202 if (!initCompositeField(FieldToInit,
Init))
2208 return this->emitFinishInit(E);
2213 Ctx.getASTContext().getAsConstantArrayType(QT);
2216 if (
Initializing && !this->emitCheckArrayDestSize(NumElems, E))
2219 if (
Inits.size() == 1 && QT ==
Inits[0]->getType())
2223 unsigned ElementIndex = 0;
2225 if (
const auto *EmbedS =
2226 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
2234 if (!this->emitCastIntegralFloating(
classifyPrim(IL), Sem,
2235 getFPOptions(E), E))
2241 return this->emitInitElem(TargetT, ElemIndex, IL);
2243 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
2259 for (; ElementIndex != NumElems; ++ElementIndex) {
2265 return this->emitFinishInit(E);
2269 unsigned NumInits =
Inits.size();
2274 QualType ElemQT = ComplexTy->getElementType();
2276 if (NumInits == 0) {
2278 for (
unsigned I = 0; I < 2; ++I) {
2279 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2281 if (!this->emitInitElem(ElemT, I, E))
2284 }
else if (NumInits == 2) {
2285 unsigned InitIndex = 0;
2290 if (!this->emitInitElem(ElemT, InitIndex, E))
2299 unsigned NumVecElements = VecT->getNumElements();
2300 assert(NumVecElements >=
Inits.size());
2302 QualType ElemQT = VecT->getElementType();
2306 unsigned InitIndex = 0;
2313 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
2314 if (!this->emitCopyArray(ElemT, 0, InitIndex,
2315 InitVecT->getNumElements(), E))
2317 InitIndex += InitVecT->getNumElements();
2319 if (!this->emitInitElem(ElemT, InitIndex, E))
2325 assert(InitIndex <= NumVecElements);
2328 for (; InitIndex != NumVecElements; ++InitIndex) {
2329 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2331 if (!this->emitInitElem(ElemT, InitIndex, E))
2338 unsigned NumElems = MT->getNumElementsFlattened();
2339 assert(
Inits.size() == NumElems);
2341 QualType ElemQT = MT->getElementType();
2347 for (
unsigned I = 0; I != NumElems; ++I) {
2350 if (!this->emitInitElem(ElemT, I, E))
2361template <
class Emitter>
2368 return this->emitInitElem(*InitT, ElemIndex,
Init);
2374 if (!this->emitConstUint32(ElemIndex,
Init))
2376 if (!this->emitArrayElemPtrUint32(
Init))
2381template <
class Emitter>
2384 bool Activate,
bool IsOperatorCall) {
2386 llvm::BitVector NonNullArgs;
2387 if (FuncDecl && FuncDecl->
hasAttr<NonNullAttr>())
2390 bool ExplicitMemberFn =
false;
2391 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl))
2392 ExplicitMemberFn = MD->isExplicitObjectMemberFunction();
2394 unsigned ArgIndex = 0;
2395 for (
const Expr *Arg : Args) {
2397 if (!this->
visit(Arg))
2405 unsigned DeclIndex = ArgIndex - IsOperatorCall + ExplicitMemberFn;
2406 if (DeclIndex < FuncDecl->getNumParams())
2407 Source = FuncDecl->
getParamDecl(ArgIndex - IsOperatorCall +
2416 if (!this->emitGetPtrLocal(*LocalIndex, Arg))
2424 if (!this->emitActivate(Arg))
2428 if (!NonNullArgs.empty() && NonNullArgs[ArgIndex]) {
2431 if (!this->emitCheckNonNullArg(ArgT, Arg))
2442template <
class Emitter>
2447template <
class Emitter>
2453template <
class Emitter>
2459template <
class Emitter>
2477template <
class Emitter>
2479 auto It = E->
begin();
2480 return this->
visit(*It);
2485 bool AlignOfReturnsPreferred =
2492 T = Ref->getPointeeType();
2494 if (T.getQualifiers().hasUnaligned())
2500 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2506template <
class Emitter>
2513 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2519 ArgType = Ref->getPointeeType();
2525 if (
ArgType->isDependentType() || !
ArgType->isConstantSizeType())
2526 return this->emitInvalid(E);
2528 if (Kind == UETT_SizeOf)
2537 return this->emitConst(Size.getQuantity(), E);
2540 if (Kind == UETT_CountOf) {
2546 if (
const auto *CAT =
2550 return this->emitConst(CAT->getSize(), E);
2560 if (VAT->getElementType()->isArrayType()) {
2561 std::optional<APSInt> Res =
2563 ? VAT->getSizeExpr()->getIntegerConstantExpr(ASTCtx)
2568 return this->emitConst(*Res, E);
2573 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2593 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2596 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2606 return this->emitConst(Size.getQuantity(), E);
2609 if (Kind == UETT_VectorElements) {
2614 return this->emitConst(VT->getNumElements(), E);
2616 return this->emitSizelessVectorElementSize(E);
2619 if (Kind == UETT_VecStep) {
2621 unsigned N = VT->getNumElements();
2628 return this->emitConst(N, E);
2630 return this->emitConst(1, E);
2633 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2642 if (Kind == UETT_PtrAuthTypeDiscriminator) {
2644 return this->emitInvalid(E);
2646 return this->emitConst(
2647 const_cast<ASTContext &
>(ASTCtx).getPointerAuthTypeDiscriminator(
2655template <
class Emitter>
2664 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2667 if (
auto GlobalIndex =
P.getGlobal(VD)) {
2668 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
2670 if (
Member->getType()->isReferenceType())
2671 return this->emitLoadPopPtr(E);
2680 if (
const auto *MD = dyn_cast<CXXMethodDecl>(
Member);
2681 MD && !MD->isStatic()) {
2685 if (!this->
discard(Base) && !this->emitSideEffect(E))
2700 const Record::Field *F = R->getField(FD);
2704 const auto maybeLoadValue = [&]() ->
bool {
2708 return this->emitLoadPop(*T, E);
2713 if (F->Decl->getType()->isReferenceType())
2714 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2715 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2718template <
class Emitter>
2728template <
class Emitter>
2742 if (!this->
visit(Common))
2744 return this->emitCopyArray(*SubExprT, 0, 0, Size, E);
2758 for (
size_t I = 0; I != Size; ++I) {
2770template <
class Emitter>
2785 return this->emitGetLocal(SubExprT, It->second, E);
2788 if (!this->
visit(SourceExpr))
2795 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2804 return this->emitGetLocal(SubExprT, LocalIndex, E);
2808template <
class Emitter>
2821 bool IsBcpCall =
false;
2822 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2823 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2827 LabelTy LabelEnd = this->getLabel();
2828 LabelTy LabelFalse = this->getLabel();
2831 if (!this->emitPushIgnoreDiags(E))
2839 if (this->checkingForUndefinedBehavior()) {
2842 if (!this->
discard(FalseExpr))
2859 if (!this->jumpFalse(LabelFalse, E))
2864 if (!this->jump(LabelEnd, E))
2866 this->emitLabel(LabelFalse);
2870 this->fallthrough(LabelEnd);
2871 this->emitLabel(LabelEnd);
2874 return this->emitPopIgnoreDiags(E);
2878template <
class Emitter>
2884 unsigned StringIndex =
P.createGlobalString(E);
2885 return this->emitGetPtrGlobal(StringIndex, E);
2890 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2891 assert(CAT &&
"a string literal that's not a constant array?");
2896 unsigned N = std::min(ArraySize, E->
getLength());
2899 for (
unsigned I = 0; I != N; ++I) {
2902 if (CharWidth == 1) {
2903 this->emitConstSint8(CodeUnit, E);
2904 this->emitInitElemSint8(I, E);
2905 }
else if (CharWidth == 2) {
2906 this->emitConstUint16(CodeUnit, E);
2907 this->emitInitElemUint16(I, E);
2908 }
else if (CharWidth == 4) {
2909 this->emitConstUint32(CodeUnit, E);
2910 this->emitInitElemUint32(I, E);
2912 llvm_unreachable(
"unsupported character width");
2917 for (
unsigned I = N; I != ArraySize; ++I) {
2918 if (CharWidth == 1) {
2919 this->emitConstSint8(0, E);
2920 this->emitInitElemSint8(I, E);
2921 }
else if (CharWidth == 2) {
2922 this->emitConstUint16(0, E);
2923 this->emitInitElemUint16(I, E);
2924 }
else if (CharWidth == 4) {
2925 this->emitConstUint32(0, E);
2926 this->emitInitElemUint32(I, E);
2928 llvm_unreachable(
"unsupported character width");
2935template <
class Emitter>
2939 return this->emitDummyPtr(E, E);
2942template <
class Emitter>
2944 auto &A =
Ctx.getASTContext();
2953template <
class Emitter>
2961 auto &A =
Ctx.getASTContext();
2965 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2966 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2973 unsigned StringIndex =
P.createGlobalString(SL);
2974 return this->emitGetPtrGlobal(StringIndex, E);
2977template <
class Emitter>
2981 return this->emitConst(E->
getValue(), E);
2984template <
class Emitter>
3010 if (!this->emitSetLocal(*RT, TempOffset, E))
3016 if (!this->emitLoad(LHST, E))
3020 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
3021 LHSComputationType, E))
3025 if (!this->emitGetLocal(*RT, TempOffset, E))
3030 if (!this->emitAddf(getFPOptions(E), E))
3034 if (!this->emitSubf(getFPOptions(E), E))
3038 if (!this->emitMulf(getFPOptions(E), E))
3042 if (!this->emitDivf(getFPOptions(E), E))
3053 return this->emitStorePop(LHST, E);
3054 return this->emitStore(LHST, E);
3057template <
class Emitter>
3066 if (Op != BO_AddAssign && Op != BO_SubAssign)
3075 if (!this->emitLoad(*
LT, LHS))
3081 if (Op == BO_AddAssign) {
3082 if (!this->emitAddOffset(*RT, E))
3085 if (!this->emitSubOffset(*RT, E))
3090 return this->emitStorePopPtr(E);
3091 return this->emitStorePtr(E);
3094template <
class Emitter>
3107 if (!
Ctx.getLangOpts().CPlusPlus14)
3108 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
3110 if (!
LT || !RT || !ResultT || !LHSComputationT)
3135 if (!this->emitSetLocal(*RT, TempOffset, E))
3142 if (!this->emitLoad(*
LT, E))
3144 if (
LT != LHSComputationT &&
3150 if (!this->emitGetLocal(*RT, TempOffset, E))
3156 if (!this->emitAdd(*LHSComputationT, E))
3160 if (!this->emitSub(*LHSComputationT, E))
3164 if (!this->emitMul(*LHSComputationT, E))
3168 if (!this->emitDiv(*LHSComputationT, E))
3172 if (!this->emitRem(*LHSComputationT, E))
3176 if (!this->emitShl(*LHSComputationT, *RT, E))
3180 if (!this->emitShr(*LHSComputationT, *RT, E))
3184 if (!this->emitBitAnd(*LHSComputationT, E))
3188 if (!this->emitBitXor(*LHSComputationT, E))
3192 if (!this->emitBitOr(*LHSComputationT, E))
3196 llvm_unreachable(
"Unimplemented compound assign operator");
3200 if (ResultT != LHSComputationT &&
3201 !this->emitIntegralCast(*LHSComputationT, *ResultT, E->
getType(), E))
3207 return this->emitStoreBitFieldPop(*ResultT, E);
3208 return this->emitStorePop(*ResultT, E);
3211 return this->emitStoreBitField(*ResultT, E);
3212 return this->emitStore(*ResultT, E);
3215template <
class Emitter>
3223template <
class Emitter>
3238 if (!
Ctx.getLangOpts().CPlusPlus11)
3245 for (
const Expr *LHS : CommaLHSs) {
3267 if (!this->
visit(Inner))
3272 if (!this->emitInitGlobalTemp(*InnerT, *GlobalIndex, TempDecl, E))
3275 if (!this->emitInitGlobal(*InnerT, *GlobalIndex, E))
3278 return this->emitGetPtrGlobal(*GlobalIndex, E);
3281 if (!this->checkLiteralType(Inner))
3284 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3290 return this->emitInitGlobalTempComp(TempDecl, E);
3303 unsigned LocalIndex =
3305 if (!this->VarScope->LocalsAlwaysEnabled &&
3306 !this->emitEnableLocal(LocalIndex, E))
3309 if (!this->
visit(Inner))
3311 if (!this->emitSetLocal(*InnerT, LocalIndex, E))
3314 return this->emitGetPtrLocal(LocalIndex, E);
3317 if (!this->checkLiteralType(Inner))
3324 if (!this->VarScope->LocalsAlwaysEnabled &&
3325 !this->emitEnableLocal(*LocalIndex, E))
3328 if (!this->emitGetPtrLocal(*LocalIndex, E))
3335template <
class Emitter>
3346 if (!this->
visit(SubExpr))
3350 return this->emitPopPtr(E);
3354template <
class Emitter>
3375 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3380 if (
P.isGlobalInitialized(*GlobalIndex))
3386 return this->emitInitGlobal(*T, *GlobalIndex, E);
3398 unsigned LocalIndex;
3402 LocalIndex = *MaybeIndex;
3406 if (!this->emitGetPtrLocal(LocalIndex, E))
3410 return this->
visit(
Init) && this->emitInit(*T, E);
3414template <
class Emitter>
3427template <
class Emitter>
3431 return this->emitConst(E->
getValue(), E);
3434template <
class Emitter>
3447 for (
const Record::Field &F : R->fields()) {
3449 if (!
Init ||
Init->containsErrors())
3457 if (!this->emitInitField(*T, F.Offset, E))
3460 if (!this->emitGetPtrField(F.Offset, E))
3471template <
class Emitter>
3478 return this->emitGetPtrGlobal(StringIndex, E);
3484template <
class Emitter>
3489 return this->emitInvalid(E);
3492template <
class Emitter>
3518 if (PointeeToT && PointeeFromT) {
3537 bool Fatal = (ToT != FromT);
3544template <
class Emitter>
3547 if (!
Ctx.getLangOpts().CPlusPlus20) {
3555 if (!this->emitCheckDynamicCast(E))
3559 return this->emitPopPtr(E);
3563template <
class Emitter>
3569 return this->emitConstBool(E->
getValue(), E);
3572template <
class Emitter>
3577 if (T->isRecordType()) {
3591 if (!this->emitGetPtrLocal(*LocalIndex, E))
3599 T->getAsCXXRecordDecl()))
3609 if (!this->visitZeroRecordInitializer(R, E))
3621 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3623 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3624 if (!this->emitCheckFunctionDecl(Ctor, E))
3635 assert(
Func->hasThisPointer());
3636 assert(!
Func->hasRVO());
3640 if (!this->emitDupPtr(E))
3644 for (
const auto *Arg : E->
arguments()) {
3645 if (!this->
visit(Arg))
3649 if (
Func->isVariadic()) {
3650 uint32_t VarArgSize = 0;
3651 unsigned NumParams =
Func->getNumWrittenParams();
3652 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3656 if (!this->emitCallVar(
Func, VarArgSize, E))
3659 if (!this->emitCall(
Func, 0, E)) {
3664 (void)this->emitPopPtr(E);
3670 return this->emitPopPtr(E);
3674 if (T->isArrayType()) {
3679 if (!this->emitDupPtr(E))
3683 initArrayDimension = [&](
QualType T) ->
bool {
3684 if (!T->isArrayType()) {
3686 for (
const auto *Arg : E->
arguments()) {
3687 if (!this->
visit(Arg))
3691 return this->emitCall(
Func, 0, E);
3695 Ctx.getASTContext().getAsConstantArrayType(T);
3700 for (
size_t I = 0; I != NumElems; ++I) {
3701 if (!this->emitConstUint64(I, E))
3703 if (!this->emitArrayElemPtrUint64(E))
3705 if (!initArrayDimension(ElemTy))
3708 return this->emitPopPtr(E);
3711 return initArrayDimension(E->
getType());
3717template <
class Emitter>
3727 assert(Val.
isInt());
3729 return this->emitConst(I, E);
3736 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3737 return this->
visit(LValueExpr);
3752 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3756 const APValue &
V = UGCD->getValue();
3757 for (
unsigned I = 0, N = R->getNumFields(); I != N; ++I) {
3758 const Record::Field *F = R->getField(I);
3759 const APValue &FieldValue =
V.getStructField(I);
3765 if (!this->emitInitField(FieldT, F->Offset, E))
3773template <
class Emitter>
3779 for (
unsigned I = 0; I != N; ++I) {
3786 if (!this->
discard(ArrayIndexExpr))
3791 if (!this->
visit(ArrayIndexExpr))
3795 if (!this->emitCast(IndexT,
PT_Sint64, E))
3805 return this->emitOffsetOf(T, E, E);
3808template <
class Emitter>
3817 return this->visitZeroInitializer(*T, Ty, E);
3824 if (!this->emitGetPtrLocal(*LocalIndex, E))
3832 ElemQT = CT->getElementType();
3835 NumElems = VT->getNumElements();
3836 ElemQT = VT->getElementType();
3842 for (
unsigned I = 0, N = NumElems; I != N; ++I) {
3843 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3845 if (!this->emitInitElem(ElemT, I, E))
3854template <
class Emitter>
3859template <
class Emitter>
3865template <
class Emitter>
3870template <
class Emitter>
3875 return this->emitConst(E->
getValue(), E);
3878template <
class Emitter>
3883 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3897 unsigned ParamIndex = 0;
3901 if (!this->emitGetParam(PT, ParamIndex, E))
3906 return this->emitCall(F, 0, E);
3911template <
class Emitter>
3919 const Expr *PlacementDest =
nullptr;
3920 bool IsNoThrow =
false;
3925 if (PlacementArgs != 0) {
3934 if (PlacementArgs == 1) {
3942 if (!this->emitInvalidNewDeleteExpr(E, E))
3947 if (OperatorNew->isReservedGlobalPlacementOperator())
3948 PlacementDest = Arg1;
3952 return this->emitInvalid(E);
3954 }
else if (!OperatorNew
3955 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3956 return this->emitInvalidNewDeleteExpr(E, E);
3959 if (!PlacementDest) {
3964 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3967 Desc =
P.createDescriptor(
3970 false,
false,
false,
3976 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3980 const Expr *Stripped = *ArraySizeExpr;
3981 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3982 Stripped = ICE->getSubExpr())
3983 if (ICE->getCastKind() != CK_NoOp &&
3984 ICE->getCastKind() != CK_IntegralCast)
3992 if (!this->
visit(Stripped))
3994 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
3997 if (PlacementDest) {
3998 if (!this->
visit(PlacementDest))
4000 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4002 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
4005 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4010 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
4014 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
4021 size_t StaticInitElems = 0;
4022 const Expr *DynamicInit =
nullptr;
4026 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
4027 StaticInitElems = CAT->getZExtSize();
4032 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init)) {
4033 if (ILE->hasArrayFiller())
4034 DynamicInit = ILE->getArrayFiller();
4053 const Function *CtorFunc =
nullptr;
4054 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
4058 }
else if (!DynamicInit && !ElemT)
4061 LabelTy EndLabel = this->getLabel();
4062 LabelTy StartLabel = this->getLabel();
4067 if (!this->emitDupPtr(E))
4069 if (!this->emitNullPtr(0,
nullptr, E))
4071 if (!this->emitEQPtr(E))
4073 if (!this->jumpTrue(EndLabel, E))
4080 if (!this->emitConst(StaticInitElems,
SizeT, E))
4082 if (!this->emitSetLocal(
SizeT, Iter, E))
4085 this->fallthrough(StartLabel);
4086 this->emitLabel(StartLabel);
4088 if (!this->emitGetLocal(
SizeT, Iter, E))
4090 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4092 if (!this->emitLT(
SizeT, E))
4094 if (!this->jumpFalse(EndLabel, E))
4098 if (!this->emitGetLocal(
SizeT, Iter, E))
4100 if (!this->emitArrayElemPtr(
SizeT, E))
4103 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
4108 if (!this->visitZeroInitializer(InitT, ElemType, E))
4110 if (!this->emitStorePop(InitT, E))
4112 }
else if (DynamicInit) {
4114 if (!this->
visit(DynamicInit))
4116 if (!this->emitStorePop(*InitT, E))
4123 if (!this->visitZeroInitializer(
4127 if (!this->emitStorePop(*ElemT, E))
4131 if (!this->emitCall(CtorFunc, 0, E))
4136 if (!this->emitGetPtrLocal(Iter, E))
4138 if (!this->emitIncPop(
SizeT,
false, E))
4141 if (!this->jump(StartLabel, E))
4144 this->fallthrough(EndLabel);
4145 this->emitLabel(EndLabel);
4149 if (PlacementDest) {
4150 if (!this->
visit(PlacementDest))
4152 if (!this->emitCheckNewTypeMismatch(E, E))
4157 if (!this->emitAlloc(Desc, E))
4166 if (!this->emitInit(*ElemT, E))
4177 return this->emitPopPtr(E);
4182template <
class Emitter>
4188 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
4189 return this->emitInvalidNewDeleteExpr(E, E);
4198template <
class Emitter>
4204 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
4209 return this->emitGetFnPtr(
Func, E);
4212template <
class Emitter>
4216 auto canonType = [](
const Type *T) {
4217 return T->getCanonicalTypeUnqualified().getTypePtr();
4225 return this->emitGetTypeid(
4229 return this->emitGetTypeid(
4238 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
4244 if (!this->emitGetTypeidPtr(TypeInfoType, E))
4247 return this->emitPopPtr(E);
4251template <
class Emitter>
4255 return this->emitDummyPtr(E, E);
4256 return this->emitError(E);
4259template <
class Emitter>
4262 return this->emitDummyPtr(E, E);
4263 return this->emitError(E);
4266template <
class Emitter>
4268 assert(
Ctx.getLangOpts().CPlusPlus);
4269 return this->emitConstBool(E->
getValue(), E);
4272template <
class Emitter>
4284 return this->emitDummyPtr(GuidDecl, E);
4289 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
4298 assert(
V.isStruct());
4299 assert(
V.getStructNumBases() == 0);
4303 return this->emitFinishInit(E);
4306template <
class Emitter>
4316template <
class Emitter>
4325template <
class Emitter>
4331template <
class Emitter>
4335 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
4339 if (OVE->isUnique())
4355template <
class Emitter>
4360template <
class Emitter>
4362 return this->emitError(E);
4365template <
class Emitter>
4371 return this->emitDummyPtr(E, E);
4374template <
class Emitter>
4375bool Compiler<Emitter>::emitVectorConversion(
const Expr *Src,
const Expr *E) {
4380 QualType ElemType = VT->getElementType();
4381 PrimType ElemT = classifyPrim(ElemType);
4383 PrimType SrcElemT = classifyVectorElementType(SrcType);
4389 if (!this->emitGetPtrLocal(*LocalIndex, E))
4393 unsigned SrcOffset =
4394 this->allocateLocalPrimitive(Src,
PT_Ptr,
true);
4395 if (!this->visit(Src))
4397 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4400 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4401 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4403 if (!this->emitArrayElemPop(SrcElemT, I, E))
4407 if (SrcElemT != ElemT) {
4408 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4410 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4411 const auto *TargetSemantics = &Ctx.getFloatSemantics(ElemType);
4415 if (!this->emitInitElem(ElemT, I, E))
4421template <
class Emitter>
4423 return emitVectorConversion(E->
getSrcExpr(), E);
4426template <
class Emitter>
4430 return this->emitInvalid(E);
4439 assert(NumOutputElems > 0);
4445 if (!this->emitGetPtrLocal(*LocalIndex, E))
4450 unsigned VectorOffsets[2];
4451 for (
unsigned I = 0; I != 2; ++I) {
4454 if (!this->
visit(Vecs[I]))
4456 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4459 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4461 assert(ShuffleIndex >= -1);
4462 if (ShuffleIndex == -1)
4463 return this->emitInvalidShuffleVectorIndex(I, E);
4465 assert(ShuffleIndex < (NumInputElems * 2));
4466 if (!this->emitGetLocal(
PT_Ptr,
4467 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4469 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4470 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4473 if (!this->emitInitElem(ElemT, I, E))
4478 return this->emitPopPtr(E);
4483template <
class Emitter>
4488 Base->getType()->isVectorType() ||
4494 if (Indices.size() == 1) {
4499 if (!this->emitConstUint32(Indices[0], E))
4501 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4511 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4519 if (!this->emitGetPtrLocal(*ResultIndex, E))
4527 uint32_t DstIndex = 0;
4528 for (uint32_t I : Indices) {
4529 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4531 if (!this->emitArrayElemPop(ElemT, I, E))
4533 if (!this->emitInitElem(ElemT, DstIndex, E))
4543template <
class Emitter>
4547 return this->
discard(SubExpr) && this->emitInvalid(E);
4553 return this->emitDummyPtr(E, E);
4556template <
class Emitter>
4561 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4566 if (!this->
visit(SubExpr))
4568 if (!this->emitConstUint8(0, E))
4570 if (!this->emitArrayElemPtrPopUint8(E))
4572 if (!this->emitInitFieldPtr(R->getField(0u)->Offset, E))
4577 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4579 return this->emitInitField(SecondFieldT, R->getField(1u)->Offset, E);
4581 assert(SecondFieldT ==
PT_Ptr);
4583 if (!this->emitGetFieldPtr(R->getField(0u)->Offset, E))
4585 if (!this->emitExpandPtr(E))
4589 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4591 return this->emitInitFieldPtr(R->getField(1u)->Offset, E);
4594template <
class Emitter>
4609 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4611 return this->emitUnsupported(E);
4620 return this->
Visit(E);
4627 return this->
Visit(E);
4631 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4634 if (
const auto *CE = dyn_cast<CastExpr>(E);
4636 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4643 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4646 if (
const auto *CE = dyn_cast<CastExpr>(E);
4647 CE && (CE->getCastKind() == CK_DerivedToBase ||
4648 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4649 CE->getCastKind() == CK_NoOp))
4669 if (!this->emitGetPtrLocal(*LocalIndex, E))
4679 return this->
Visit(E);
4682template <
class Emitter>
4688 return this->
Visit(E) && this->emitFinishInit(E);
4691template <
class Emitter>
4697 return this->
Visit(E) && this->emitFinishInitPop(E);
4703 return this->
Visit(E);
4714 if (!this->
visit(E))
4716 return this->emitComplexBoolCast(E);
4721 if (!this->
visit(E))
4729 return this->emitIsNonNullPtr(E);
4733 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4736 return this->emitCast(*T,
PT_Bool, E);
4739template <
class Emitter>
4743 QT = AT->getValueType();
4747 return this->emitZeroBool(E);
4749 return this->emitZeroSint8(E);
4751 return this->emitZeroUint8(E);
4753 return this->emitZeroSint16(E);
4755 return this->emitZeroUint16(E);
4757 return this->emitZeroSint32(E);
4759 return this->emitZeroUint32(E);
4761 return this->emitZeroSint64(E);
4763 return this->emitZeroUint64(E);
4765 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4767 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4769 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4772 return this->emitNullMemberPtr(0,
nullptr, E);
4774 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4775 return this->emitFloat(F, E);
4778 auto Sem = Ctx.getASTContext().getFixedPointSemantics(QT);
4782 llvm_unreachable(
"unknown primitive type");
4785template <
class Emitter>
4786bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4791 for (
const Record::Field &Field :
R->fields()) {
4792 if (Field.isUnnamedBitField())
4799 if (!this->visitZeroInitializer(T, QT, E))
4802 if (!this->emitInitFieldActivate(T, Field.Offset, E))
4806 if (!this->emitInitField(T, Field.Offset, E))
4811 if (!this->emitGetPtrField(Field.Offset, E))
4818 if (!this->visitZeroInitializer(T, ET, E))
4820 if (!this->emitInitElem(T, I, E))
4825 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4828 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4836 if (!this->emitFinishInitActivatePop(E))
4840 if (!this->emitFinishInitPop(E))
4844 for (
const Record::Base &B :
R->bases()) {
4845 if (!this->emitGetPtrBase(B.Offset, E))
4847 if (!this->visitZeroRecordInitializer(B.R, E))
4849 if (!this->emitFinishInitPop(E))
4858template <
class Emitter>
4859bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4860 assert(T->isArrayType() || T->isAnyComplexType() || T->isVectorType());
4866 for (
size_t I = 0; I != NumElems; ++I) {
4867 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4869 if (!this->emitInitElem(*ElemT, I, E))
4875 const Record *
R = getRecord(ElemType);
4879 for (
size_t I = 0; I != NumElems; ++I) {
4880 if (!this->emitConstUint32(I, E))
4882 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4884 if (!this->visitZeroRecordInitializer(R, E))
4886 if (!this->emitPopPtr(E))
4892 for (
size_t I = 0; I != NumElems; ++I) {
4893 if (!this->emitConstUint32(I, E))
4895 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4897 if (!this->visitZeroArrayInitializer(ElemType, E))
4899 if (!this->emitPopPtr(E))
4908template <
class Emitter>
4909bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4911 if (!canClassify(E->
getType()))
4914 if (!this->visit(RHS))
4916 if (!this->visit(LHS))
4923 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4927 bool Activates = refersToUnion(LHS);
4930 if (!this->emitFlip(
PT_Ptr, RHT, E))
4933 if (DiscardResult) {
4934 if (BitField && Activates)
4935 return this->emitStoreBitFieldActivatePop(RHT, E);
4937 return this->emitStoreBitFieldPop(RHT, E);
4939 return this->emitStoreActivatePop(RHT, E);
4941 return this->emitStorePop(RHT, E);
4944 auto maybeLoad = [&](
bool Result) ->
bool {
4950 return this->emitLoadPop(RHT, E);
4954 if (BitField && Activates)
4955 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4957 return maybeLoad(this->emitStoreBitField(RHT, E));
4959 return maybeLoad(this->emitStoreActivate(RHT, E));
4961 return maybeLoad(this->emitStore(RHT, E));
4964template <
class Emitter>
4965template <
typename T>
4969 return this->emitConstSint8(
Value, E);
4971 return this->emitConstUint8(
Value, E);
4973 return this->emitConstSint16(
Value, E);
4975 return this->emitConstUint16(
Value, E);
4977 return this->emitConstSint32(
Value, E);
4979 return this->emitConstUint32(
Value, E);
4981 return this->emitConstSint64(
Value, E);
4983 return this->emitConstUint64(
Value, E);
4985 return this->emitConstBool(
Value, E);
4992 llvm_unreachable(
"Invalid integral type");
4995 llvm_unreachable(
"unknown primitive type");
4998template <
class Emitter>
4999template <
typename T>
5000bool Compiler<Emitter>::emitConst(T
Value,
const Expr *E) {
5001 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5004template <
class Emitter>
5008 return this->emitConstIntAPS(
Value, E);
5010 return this->emitConstIntAP(
Value, E);
5012 if (
Value.isSigned())
5013 return this->emitConst(
Value.getSExtValue(), Ty, E);
5014 return this->emitConst(
Value.getZExtValue(), Ty, E);
5017template <
class Emitter>
5021 return this->emitConstIntAPS(
Value, E);
5023 return this->emitConstIntAP(
Value, E);
5026 return this->emitConst(
Value.getSExtValue(), Ty, E);
5027 return this->emitConst(
Value.getZExtValue(), Ty, E);
5030template <
class Emitter>
5031bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
5032 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5035template <
class Emitter>
5048 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
5049 Locals.insert({VD, Local});
5050 VarScope->addForScopeKind(Local, SC);
5051 return Local.Offset;
5054template <
class Emitter>
5059 bool IsTemporary =
false;
5060 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
5063 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
5064 Init = VarD->getInit();
5066 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
5077 return std::nullopt;
5082 Locals.insert({Key, Local});
5083 VarScope->addForScopeKind(Local, SC);
5084 return Local.Offset;
5087template <
class Emitter>
5097 return std::nullopt;
5107 return Local.Offset;
5110template <
class Emitter>
5112 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
5113 return PT->getPointeeType()->getAsCanonical<RecordType>();
5119 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
5123template <
class Emitter>
5125 return P.getOrCreateRecord(RD);
5128template <
class Emitter>
5130 return Ctx.getOrCreateFunction(FD);
5133template <
class Emitter>
5137 auto maybeDestroyLocals = [&]() ->
bool {
5138 if (DestroyToplevelScope)
5139 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
5140 return this->emitCheckAllocations(E);
5147 return this->emitRetVoid(E) && maybeDestroyLocals();
5155 return this->emitRet(*T, E) && maybeDestroyLocals();
5163 if (!this->emitGetPtrLocal(*LocalOffset, E))
5171 return this->emitRetValue(E) && maybeDestroyLocals();
5174 return maybeDestroyLocals() &&
false;
5177template <
class Emitter>
5179 bool DestroyToplevelScope) {
5183 return this->
visitExpr(E, DestroyToplevelScope);
5186template <
class Emitter>
5198 if (
auto GlobalIndex =
P.getGlobal(VD)) {
5199 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5213template <
class Emitter>
5215 bool ConstantContext) {
5218 if (!ConstantContext) {
5232 auto GlobalIndex =
P.getGlobal(VD);
5233 assert(GlobalIndex);
5235 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
5238 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
5242 auto Local =
Locals.find(VD);
5243 assert(Local !=
Locals.end());
5245 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
5248 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
5258 auto GlobalIndex =
P.getGlobal(VD);
5259 assert(GlobalIndex);
5260 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5269 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
5272template <
class Emitter>
5283 if (!this->isActive())
5288 if (
Init &&
Init->isValueDependent())
5292 auto checkDecl = [&]() ->
bool {
5294 return !NeedsOp || this->emitCheckDecl(VD, VD);
5301 if (!
P.getGlobal(*GlobalIndex)->isInitialized())
5304 if (
P.isGlobalInitialized(*GlobalIndex))
5308 }
else if ((GlobalIndex =
5323 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
5326 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
5329 if (!this->emitStartInit(
Init))
5335 if (!this->emitEndInit(
Init))
5338 return this->emitFinishInitGlobal(
Init);
5348 if (!
Init ||
Init->getType()->isVoidType())
5357 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
5369 if (!this->emitCheckRefInit(
Init))
5373 return this->emitSetLocal(*VarT, Offset, VD);
5381 if (!this->emitGetPtrLocal(*Offset,
Init))
5389template <
class Emitter>
5409 Locals.insert({VD, Local});
5412 if (!this->emitGetPtrLocal(Local.Offset, VD))
5416 return this->emitDestructionPop(D, VD);
5419template <
class Emitter>
5424 return this->emitConst(Val.
getInt(), ValType, E);
5427 return this->emitFloat(F, E);
5432 if (!this->emitGetMemberPtr(MemberDecl, E))
5438 if (!this->emitCopyMemberPtrPath(PathEntry, IsDerived, E))
5444 return this->emitNullMemberPtr(0,
nullptr, E);
5449 return this->emitNull(ValType, 0,
nullptr, E);
5454 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
5455 return this->
visit(BaseExpr);
5460 QualType EntryType = VD->getType();
5461 for (
auto &Entry : Path) {
5463 uint64_t Index = Entry.getAsArrayIndex();
5466 if (!this->emitConst(Index,
PT_Uint64, E))
5468 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
5470 EntryType = ElemType;
5480 const Decl *BaseOrMember = Entry.getAsBaseOrMember().getPointer();
5481 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
5483 if (!this->emitGetPtrFieldPop(EntryOffset, E))
5485 EntryType = FD->getType();
5489 if (!this->emitGetPtrBasePop(BaseOffset,
false, E))
5491 EntryType =
Ctx.getASTContext().getCanonicalTagType(
Base);
5503template <
class Emitter>
5511 const Record::Field *RF = R->getField(I);
5512 QualType FieldType = RF->Decl->getType();
5518 if (!this->emitInitField(*PT, RF->Offset, E))
5521 if (!this->emitGetPtrField(RF->Offset, E))
5525 if (!this->emitFinishInitPop(E))
5533 const Record::Base *RB = R->getBase(I);
5534 QualType BaseType =
Ctx.getASTContext().getCanonicalTagType(RB->Decl);
5536 if (!this->emitGetPtrBase(RB->Offset, E))
5540 if (!this->emitFinishInitPop(E))
5553 const Record::Field *RF = R->getField(UnionField);
5554 QualType FieldType = RF->Decl->getType();
5559 if (RF->isBitField())
5560 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
5562 return this->emitInitFieldActivate(*PT, RF->Offset, E);
5565 if (!this->emitGetPtrField(RF->Offset, E))
5567 if (!this->emitActivate(E))
5571 return this->emitPopPtr(E);
5575 const auto *ArrType = T->getAsArrayTypeUnsafe();
5576 QualType ElemType = ArrType->getElementType();
5579 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5580 const APValue &Elem = A >= InitializedElems
5587 if (!this->emitInitElem(*ElemT, A, E))
5590 if (!this->emitConstUint32(A, E))
5592 if (!this->emitArrayElemPtrUint32(E))
5596 if (!this->emitPopPtr(E))
5607template <
class Emitter>
5609 unsigned BuiltinID) {
5610 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5615 return this->emitConst(0, E);
5618 if (!this->emitStartSpeculation(E))
5620 LabelTy EndLabel = this->getLabel();
5621 if (!this->speculate(E, EndLabel))
5623 if (!this->emitEndSpeculation(E))
5625 this->fallthrough(EndLabel);
5633 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5634 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5635 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5636 BuiltinID == Builtin::BI__builtin_function_start) {
5639 return this->emitDummyPtr(E, E);
5650 if (!this->emitGetPtrLocal(*LocalIndex, E))
5655 switch (BuiltinID) {
5656 case Builtin::BI__builtin_object_size:
5657 case Builtin::BI__builtin_dynamic_object_size: {
5661 if (!this->
visit(Arg0))
5672 case Builtin::BI__assume:
5673 case Builtin::BI__builtin_assume:
5676 case Builtin::BI__atomic_is_lock_free:
5677 case Builtin::BI__atomic_always_lock_free: {
5688 for (
const auto *Arg : E->
arguments()) {
5689 if (!this->
visit(Arg))
5695 if (!this->emitCallBI(E, BuiltinID, E))
5704template <
class Emitter>
5725 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5726 DD && DD->isTrivial()) {
5728 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5730 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5731 this->emitPopPtr(E);
5739 bool HasRVO = !
ReturnType->isVoidType() && !T;
5747 if (!this->emitGetPtrLocal(*LocalIndex, E))
5755 if (!this->emitGetPtrLocal(*LocalIndex, E))
5759 if (!this->emitDupPtr(E))
5766 bool IsAssignmentOperatorCall =
false;
5767 bool ActivateLHS =
false;
5768 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5769 OCE && OCE->isAssignmentOp()) {
5773 assert(Args.size() == 2);
5774 const CXXRecordDecl *LHSRecord = Args[0]->getType()->getAsCXXRecordDecl();
5776 IsAssignmentOperatorCall =
true;
5777 std::reverse(Args.begin(), Args.end());
5783 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5784 MD && MD->isStatic()) {
5788 Args.erase(Args.begin());
5792 bool Devirtualized =
false;
5795 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5803 if (!this->
visit(Callee))
5805 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5807 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5809 if (!this->emitGetMemberPtrBase(E))
5812 const auto *InstancePtr = MC->getImplicitObjectArgument();
5819 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5820 Devirtualized =
true;
5821 if (!this->
visit(Stripped))
5824 if (!this->
visit(InstancePtr))
5828 if (!this->
visit(InstancePtr))
5832 }
else if (
const auto *PD =
5833 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5834 if (!this->emitCheckPseudoDtor(E))
5842 return this->emitEndLifetimePop(E);
5843 }
else if (!FuncDecl) {
5847 if (!this->
visit(Callee))
5849 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5858 if (IsAssignmentOperatorCall) {
5859 assert(Args.size() == 2);
5862 if (!this->emitFlip(Arg2T, Arg1T, E))
5876 assert(HasRVO ==
Func->hasRVO());
5878 bool HasQualifier =
false;
5879 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5880 HasQualifier = ME->hasQualifier();
5882 bool IsVirtual =
false;
5883 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5884 IsVirtual = !Devirtualized && MD->isVirtual();
5889 if (IsVirtual && !HasQualifier) {
5890 uint32_t VarArgSize = 0;
5891 unsigned NumParams =
5892 Func->getNumWrittenParams() +
5894 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5897 if (!this->emitCallVirt(
Func, VarArgSize, E))
5899 }
else if (
Func->isVariadic()) {
5900 uint32_t VarArgSize = 0;
5901 unsigned NumParams =
5902 Func->getNumWrittenParams() +
5904 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5906 if (!this->emitCallVar(
Func, VarArgSize, E))
5909 if (!this->emitCall(
Func, 0, E))
5918 uint32_t ArgSize = 0;
5919 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5925 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5927 if (!this->emitGetMemberPtrDecl(E))
5930 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5933 if (!this->emitCallPtr(ArgSize, E, E))
5944template <
class Emitter>
5951template <
class Emitter>
5958template <
class Emitter>
5963 return this->emitConstBool(E->
getValue(), E);
5966template <
class Emitter>
5972 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5973 return this->emitNullPtr(Val,
nullptr, E);
5976template <
class Emitter>
5984 return this->emitZero(T, E);
5987template <
class Emitter>
5992 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
5993 if (this->LambdaThisCapture.Offset > 0) {
5994 if (this->LambdaThisCapture.IsPtr)
5995 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
5996 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
6005 return this->emitThis(E);
6020 unsigned StartIndex = 0;
6021 unsigned EndIndex = 0;
6023 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
6025 EndIndex = StartIndex;
6032 for (; StartIndex > 0; --StartIndex) {
6042 if (StartIndex == 0 && EndIndex == 0)
6045 assert(StartIndex < EndIndex);
6048 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
6060 case Stmt::CompoundStmtClass:
6062 case Stmt::DeclStmtClass:
6064 case Stmt::ReturnStmtClass:
6066 case Stmt::IfStmtClass:
6068 case Stmt::WhileStmtClass:
6070 case Stmt::DoStmtClass:
6072 case Stmt::ForStmtClass:
6074 case Stmt::CXXForRangeStmtClass:
6076 case Stmt::BreakStmtClass:
6078 case Stmt::ContinueStmtClass:
6080 case Stmt::SwitchStmtClass:
6082 case Stmt::CaseStmtClass:
6084 case Stmt::DefaultStmtClass:
6086 case Stmt::AttributedStmtClass:
6088 case Stmt::CXXTryStmtClass:
6090 case Stmt::NullStmtClass:
6093 case Stmt::GCCAsmStmtClass:
6094 case Stmt::MSAsmStmtClass:
6095 case Stmt::GotoStmtClass:
6096 return this->emitInvalid(S);
6097 case Stmt::LabelStmtClass:
6100 if (
const auto *E = dyn_cast<Expr>(S))
6107template <
class Emitter>
6110 for (
const auto *InnerStmt : S->
body())
6113 return Scope.destroyLocals();
6116template <
class Emitter>
6117bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
6118 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
6119 for (
auto *BD : DD->flat_bindings())
6120 if (
auto *KD = BD->getHoldingVar();
6121 KD && !this->visitVarDecl(KD, KD->getInit()))
6135template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
6137 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
6138 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
6145 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
6150 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
6151 ICE && (ICE->getCastKind() == CK_NoOp ||
6152 ICE->getCastKind() == CK_DerivedToBase ||
6153 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
6154 E = ICE->getSubExpr();
6158 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
6159 const auto *ThisRecord =
6160 This->getType()->getPointeeType()->getAsRecordDecl();
6161 if (!ThisRecord->isUnion())
6164 if (
const auto *Ctor =
6165 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
6166 return Ctor->getParent() == ThisRecord;
6175template <
class Emitter>
6177 bool EvaluateConditionDecl) {
6178 for (
const auto *D : DS->
decls()) {
6183 const auto *VD = dyn_cast<VarDecl>(D);
6190 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
6197template <
class Emitter>
6200 return this->emitUnsupported(RS);
6206 if (!this->
visit(RE))
6212 if (RE->getType()->isVoidType()) {
6213 if (!this->
visit(RE))
6216 if (RE->containsErrors())
6221 if (!this->emitRVOPtr(RE))
6227 return this->emitRetVoid(RS);
6233 return this->emitRetVoid(RS);
6239 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
6246 if (
auto *CondInit = IS->
getInit()) {
6262 return visitChildStmt(IS->
getThen());
6264 return visitChildStmt(Else);
6270 if (!this->emitIsConstantContext(IS))
6273 if (!this->emitIsConstantContext(IS))
6275 if (!this->emitInv(IS))
6289 LabelTy LabelElse = this->getLabel();
6290 LabelTy LabelEnd = this->getLabel();
6291 if (!this->jumpFalse(LabelElse, IS))
6293 if (!visitChildStmt(IS->
getThen()))
6295 if (!this->jump(LabelEnd, IS))
6297 this->emitLabel(LabelElse);
6298 if (!visitChildStmt(Else))
6300 this->emitLabel(LabelEnd);
6302 LabelTy LabelEnd = this->getLabel();
6303 if (!this->jumpFalse(LabelEnd, IS))
6305 if (!visitChildStmt(IS->
getThen()))
6307 this->emitLabel(LabelEnd);
6316template <
class Emitter>
6321 LabelTy CondLabel = this->getLabel();
6322 LabelTy EndLabel = this->getLabel();
6326 this->fallthrough(CondLabel);
6327 this->emitLabel(CondLabel);
6343 if (!this->jumpFalse(EndLabel, S))
6353 if (!this->jump(CondLabel, S))
6355 this->fallthrough(EndLabel);
6356 this->emitLabel(EndLabel);
6365 LabelTy StartLabel = this->getLabel();
6366 LabelTy EndLabel = this->getLabel();
6367 LabelTy CondLabel = this->getLabel();
6371 this->fallthrough(StartLabel);
6372 this->emitLabel(StartLabel);
6378 this->fallthrough(CondLabel);
6379 this->emitLabel(CondLabel);
6386 if (!this->jumpTrue(StartLabel, S))
6389 this->fallthrough(EndLabel);
6390 this->emitLabel(EndLabel);
6394template <
class Emitter>
6402 LabelTy EndLabel = this->getLabel();
6403 LabelTy CondLabel = this->getLabel();
6404 LabelTy IncLabel = this->getLabel();
6411 this->fallthrough(CondLabel);
6412 this->emitLabel(CondLabel);
6424 if (!this->jumpFalse(EndLabel, S))
6433 this->fallthrough(IncLabel);
6434 this->emitLabel(IncLabel);
6440 if (!this->jump(CondLabel, S))
6444 this->emitLabel(EndLabel);
6450template <
class Emitter>
6460 LabelTy EndLabel = this->getLabel();
6461 LabelTy CondLabel = this->getLabel();
6462 LabelTy IncLabel = this->getLabel();
6477 this->fallthrough(CondLabel);
6478 this->emitLabel(CondLabel);
6481 if (!this->jumpFalse(EndLabel, S))
6492 this->fallthrough(IncLabel);
6493 this->emitLabel(IncLabel);
6498 if (!this->jump(CondLabel, S))
6501 this->fallthrough(EndLabel);
6502 this->emitLabel(EndLabel);
6506template <
class Emitter>
6517 if (LI.BreakLabel) {
6518 TargetLabel = *LI.BreakLabel;
6519 BreakScope = LI.BreakOrContinueScope;
6525 if (LI.Name == TargetLoop) {
6526 TargetLabel = *LI.BreakLabel;
6527 BreakScope = LI.BreakOrContinueScope;
6538 C =
C->getParent()) {
6539 if (!
C->destroyLocals())
6543 return this->jump(*TargetLabel, S);
6546template <
class Emitter>
6557 if (LI.ContinueLabel) {
6558 TargetLabel = *LI.ContinueLabel;
6559 ContinueScope = LI.BreakOrContinueScope;
6565 if (LI.Name == TargetLoop) {
6566 TargetLabel = *LI.ContinueLabel;
6567 ContinueScope = LI.BreakOrContinueScope;
6572 assert(TargetLabel);
6575 C =
C->getParent()) {
6576 if (!
C->destroyLocals())
6580 return this->jump(*TargetLabel, S);
6583template <
class Emitter>
6586 if (
Cond->containsErrors())
6592 LabelTy EndLabel = this->getLabel();
6597 if (
const auto *CondInit = S->
getInit())
6608 if (!this->emitSetLocal(CondT, CondVar, S))
6618 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6621 if (CS->caseStmtIsGNURange()) {
6622 LabelTy EndOfRangeCheck = this->getLabel();
6623 const Expr *Low = CS->getLHS();
6624 const Expr *High = CS->getRHS();
6628 if (!this->emitGetLocal(CondT, CondVar, CS))
6630 if (!this->
visit(Low))
6633 if (!this->emitGE(
LT, S))
6635 if (!this->jumpFalse(EndOfRangeCheck, S))
6638 if (!this->emitGetLocal(CondT, CondVar, CS))
6640 if (!this->
visit(High))
6643 if (!this->emitLE(HT, S))
6647 this->emitLabel(EndOfRangeCheck);
6652 if (
Value->isValueDependent())
6657 if (!this->emitGetLocal(CondT, CondVar, CS))
6663 if (!this->emitEQ(ValueT, S))
6668 assert(!DefaultLabel);
6669 DefaultLabel = this->getLabel();
6676 if (!this->jump(*DefaultLabel, S))
6679 if (!this->jump(EndLabel, S))
6687 this->fallthrough(EndLabel);
6688 this->emitLabel(EndLabel);
6693template <
class Emitter>
6700template <
class Emitter>
6707 if (LI.DefaultLabel) {
6708 DefaultLabel = *LI.DefaultLabel;
6713 this->emitLabel(DefaultLabel);
6717template <
class Emitter>
6724 if (IsMSVCConstexprAttr && !this->emitPushMSVCCE(S))
6727 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6728 !this->Ctx.getLangOpts().MSVCCompat) {
6730 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6736 const Expr *Assumption = AA->getAssumption();
6747 if (!this->emitAssume(Assumption))
6756 if (IsMSVCConstexprAttr)
6757 return this->emitPopMSVCCE(S);
6761template <
class Emitter>
6767template <
class Emitter>
6768bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6775 assert(ClosureClass->
captures().empty());
6779 "A generic lambda's static-invoker function must be a "
6780 "template specialization");
6784 void *InsertPos =
nullptr;
6785 const FunctionDecl *CorrespondingCallOpSpecialization =
6787 assert(CorrespondingCallOpSpecialization);
6788 LambdaCallOp = CorrespondingCallOpSpecialization;
6792 assert(ClosureClass->
captures().empty());
6793 const Function *
Func = this->getFunction(LambdaCallOp);
6796 assert(
Func->hasThisPointer());
6799 if (
Func->hasRVO()) {
6800 if (!this->emitRVOPtr(MD))
6808 if (!this->emitNullPtr(0,
nullptr, MD))
6813 auto It = this->Params.find(PVD);
6814 assert(It != this->Params.end());
6818 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6819 if (!this->emitGetParam(ParamType, It->second.Index, MD))
6823 if (!this->emitCall(
Func, 0, LambdaCallOp))
6828 return this->emitRet(*ReturnType, MD);
6831 return this->emitRetVoid(MD);
6834template <
class Emitter>
6835bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6836 if (Ctx.getLangOpts().CPlusPlus23)
6846 const Expr *InitExpr =
Init->getInit();
6848 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6852 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6862template <
class Emitter>
6864 assert(!ReturnType);
6867 if (!this->emitStartThisLifetime1(Ctor))
6870 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6871 const Expr *InitExpr,
6874 if (InitExpr->getType().isNull())
6878 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6881 if (!this->visit(InitExpr))
6884 if (F->isBitField())
6885 return this->emitInitThisBitField(*T, FieldOffset, F->bitWidth(),
6887 return this->emitInitThisField(*T, FieldOffset, InitExpr);
6892 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6895 if (
Activate && !this->emitActivate(InitExpr))
6898 return this->visitInitializerPop(InitExpr);
6902 const Record *
R = this->getRecord(RD);
6905 bool IsUnion =
R->isUnion();
6915 if (!this->emitThis(Ctor))
6918 if (!this->emitGetParam(
PT_Ptr, 0, Ctor))
6921 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6922 this->emitRetVoid(Ctor);
6925 unsigned FieldInits = 0;
6927 for (
const auto *
Init : Ctor->
inits()) {
6931 const Expr *InitExpr =
Init->getInit();
6933 const Record::Field *F =
R->getField(
Member);
6937 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6941 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6944 if (
Init->isBaseVirtual()) {
6945 assert(
R->getVirtualBase(BaseDecl));
6946 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6952 const Record::Base *B =
R->getBase(BaseDecl);
6954 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6958 if (IsUnion && !this->emitActivate(InitExpr))
6961 if (!this->visitInitializerPop(InitExpr))
6966 unsigned ChainSize = IFD->getChainingSize();
6967 assert(ChainSize >= 2);
6969 unsigned NestedFieldOffset = 0;
6970 const Record::Field *NestedField =
nullptr;
6971 for (
unsigned I = 0; I != ChainSize; ++I) {
6973 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6974 assert(FieldRecord);
6976 NestedField = FieldRecord->
getField(FD);
6977 assert(NestedField);
6978 IsUnion = IsUnion || FieldRecord->
isUnion();
6980 NestedFieldOffset += NestedField->Offset;
6983 if (I != ChainSize - 1)
6986 assert(NestedField);
6989 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
6994 unsigned InitFieldOffset = 0;
6995 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
6997 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6998 assert(FieldRecord);
6999 NestedField = FieldRecord->
getField(FD);
7000 InitFieldOffset += NestedField->Offset;
7001 assert(NestedField);
7002 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
7004 if (!this->emitFinishInitPop(InitExpr))
7008 InitStack.pop_back_n(ChainSize - 1);
7011 assert(
Init->isDelegatingInitializer());
7012 if (!this->emitThis(InitExpr))
7014 if (!this->visitInitializerPop(
Init->getInit()))
7018 if (!
Scope.destroyLocals())
7022 if (FieldInits !=
R->getNumFields()) {
7023 assert(FieldInits < R->getNumFields());
7025 if (!this->emitStartThisLifetime(Ctor))
7032 if (
const auto *CS = dyn_cast<CompoundStmt>(Body)) {
7033 if (!CS->body_empty() && !this->emitCtorCheck(
SourceInfo{}))
7040 if (!visitStmt(Body))
7047template <
class Emitter>
7050 const Record *
R = this->getRecord(RD);
7055 if (!this->visitStmt(Dtor->
getBody()))
7059 if (!this->emitThis(Dtor))
7062 if (!this->emitCheckDestruction(Dtor))
7066 if (!
R->isUnion()) {
7070 for (
const Record::Field &Field : llvm::reverse(
R->fields())) {
7074 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
7076 if (!this->emitDestructionPop(D,
SourceInfo{}))
7081 for (
const Record::Base &
Base : llvm::reverse(
R->bases())) {
7082 if (
Base.R->hasTrivialDtor())
7086 if (!this->emitRecordDestructionPop(
Base.R, {}))
7090 if (!this->emitMarkDestroyed(Dtor))
7094 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
7097template <
class Emitter>
7098bool Compiler<Emitter>::compileUnionAssignmentOperator(
7100 if (!this->emitThis(MD))
7103 if (!this->emitGetParam(
PT_Ptr, 0, MD))
7106 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
7109template <
class Emitter>
7119 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
7120 return this->compileConstructor(Ctor);
7121 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
7122 return this->compileDestructor(Dtor);
7125 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
7130 return this->compileUnionAssignmentOperator(MD);
7133 return this->emitLambdaStaticInvokerBody(MD);
7137 if (
const auto *Body = F->
getBody())
7151 return FD->getBitWidthValue();
7154template <
class Emitter>
7170 if (!
Ctx.getLangOpts().CPlusPlus14)
7171 return this->emitInvalid(E);
7173 return this->emitError(E);
7175 if (!this->
visit(SubExpr))
7179 if (!this->emitIncPtr(E))
7186 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
7187 : this->emitIncf(getFPOptions(E), E);
7199 if (!
Ctx.getLangOpts().CPlusPlus14)
7200 return this->emitInvalid(E);
7202 return this->emitError(E);
7204 if (!this->
visit(SubExpr))
7208 if (!this->emitDecPtr(E))
7215 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
7216 : this->emitDecf(getFPOptions(E), E);
7229 if (!
Ctx.getLangOpts().CPlusPlus14)
7230 return this->emitInvalid(E);
7232 return this->emitError(E);
7234 if (!this->
visit(SubExpr))
7238 if (!this->emitLoadPtr(E))
7240 if (!this->emitConstUint8(1, E))
7242 if (!this->emitAddOffsetUint8(E))
7244 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7250 return this->emitIncfPop(getFPOptions(E), E);
7260 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7261 if (!this->emitLoadFloat(E))
7263 APFloat F(TargetSemantics, 1);
7264 if (!this->emitFloat(F, E))
7267 if (!this->emitAddf(getFPOptions(E), E))
7269 if (!this->emitStoreFloat(E))
7281 return E->
isGLValue() || this->emitLoadPop(*T, E);
7284 if (!
Ctx.getLangOpts().CPlusPlus14)
7285 return this->emitInvalid(E);
7287 return this->emitError(E);
7289 if (!this->
visit(SubExpr))
7293 if (!this->emitLoadPtr(E))
7295 if (!this->emitConstUint8(1, E))
7297 if (!this->emitSubOffsetUint8(E))
7299 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7305 return this->emitDecfPop(getFPOptions(E), E);
7315 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7316 if (!this->emitLoadFloat(E))
7318 APFloat F(TargetSemantics, 1);
7319 if (!this->emitFloat(F, E))
7322 if (!this->emitSubf(getFPOptions(E), E))
7324 if (!this->emitStoreFloat(E))
7336 return E->
isGLValue() || this->emitLoadPop(*T, E);
7340 return this->emitError(E);
7343 return this->
discard(SubExpr);
7348 if (!this->emitInv(E))
7352 return this->emitCast(
PT_Bool, ET, E);
7356 return this->emitError(E);
7358 if (!this->
visit(SubExpr))
7360 return DiscardResult ? this->emitPop(*T, E) : this->emitNeg(*T, E);
7363 return this->emitError(E);
7365 if (!this->
visit(SubExpr))
7381 if (!
Ctx.getLangOpts().CPlusPlus) {
7383 if (
const auto *Deref = dyn_cast<UnaryOperator>(
Sub);
7384 Deref && Deref->getOpcode() == UO_Deref)
7385 return this->
delegate(Deref->getSubExpr());
7391 return this->
discard(SubExpr);
7393 if (!this->
visit(SubExpr))
7400 return this->emitNarrowPtr(E);
7405 return this->emitError(E);
7407 if (!this->
visit(SubExpr))
7409 return DiscardResult ? this->emitPop(*T, E) : this->emitComp(*T, E);
7421 : this->visitZeroInitializer(*T, SubExpr->
getType(), SubExpr);
7426 assert(
false &&
"Unhandled opcode");
7432template <
class Emitter>
7438 return this->
discard(SubExpr);
7441 auto prepareResult = [=]() ->
bool {
7446 return this->emitGetPtrLocal(*LocalIndex, E);
7453 unsigned SubExprOffset = ~0u;
7454 auto createTemp = [=, &SubExprOffset]() ->
bool {
7457 if (!this->
visit(SubExpr))
7459 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
7463 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7464 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7466 return this->emitArrayElemPop(ElemT, Index, E);
7471 if (!prepareResult())
7475 for (
unsigned I = 0; I != 2; ++I) {
7476 if (!getElem(SubExprOffset, I))
7478 if (!this->emitNeg(ElemT, E))
7480 if (!this->emitInitElem(ElemT, I, E))
7491 if (!this->
visit(SubExpr))
7493 if (!this->emitComplexBoolCast(SubExpr))
7495 if (!this->emitInv(E))
7498 return this->emitCast(
PT_Bool, ET, E);
7502 return this->emitComplexReal(SubExpr);
7505 if (!this->
visit(SubExpr))
7509 if (!this->emitConstUint8(1, E))
7511 return this->emitArrayElemPtrPopUint8(E);
7522 if (!this->emitArrayElem(ElemT, 1, E))
7524 if (!this->emitNeg(ElemT, E))
7526 if (!this->emitInitElem(ElemT, 1, E))
7534 return this->emitInvalid(E);
7540template <
class Emitter>
7546 return this->
discard(SubExpr);
7549 if (UnaryOp == UO_Extension)
7552 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
7553 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
7554 return this->emitInvalid(E);
7557 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
7564 if (!this->emitGetPtrLocal(*LocalIndex, E))
7569 unsigned SubExprOffset =
7571 if (!this->
visit(SubExpr))
7573 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
7578 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7579 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7581 return this->emitArrayElemPop(ElemT, Index, E);
7586 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7587 if (!getElem(SubExprOffset, I))
7589 if (!this->emitNeg(ElemT, E))
7591 if (!this->emitInitElem(ElemT, I, E))
7606 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7607 if (!getElem(SubExprOffset, I))
7610 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
7612 if (!this->emitInv(E))
7614 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
7616 if (!this->emitNeg(ElemT, E))
7618 if (ElemT != ResultVecElemT &&
7619 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
7621 if (!this->emitInitElem(ResultVecElemT, I, E))
7627 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7628 if (!getElem(SubExprOffset, I))
7631 if (!this->emitInv(E))
7634 if (!this->emitComp(ElemT, E))
7637 if (!this->emitInitElem(ElemT, I, E))
7642 llvm_unreachable(
"Unsupported unary operators should be handled up front");
7647template <
class Emitter>
7652 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D))
7653 return this->emitConst(ECD->getInitVal(), E);
7654 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
7656 return F && this->emitGetFnPtr(F, E);
7658 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
7663 return this->emitInitGlobal(*T, *Index, E);
7666 if (!this->emitGetPtrGlobal(*Index, E))
7670 return this->emitFinishInit(E);
7685 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
7686 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7691 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7692 if (IsReference || !It->second.IsPtr)
7693 return this->emitGetParam(
classifyPrim(E), It->second.Index, E);
7695 return this->emitGetPtrParam(It->second.Index, E);
7698 if (!
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7704 const unsigned Offset = It->second.Offset;
7707 return this->emitGetRefLocal(Offset, E);
7709 return this->emitGetPtrLocal(Offset, E);
7712 if (
auto GlobalIndex =
P.getGlobal(D)) {
7714 if (!
Ctx.getLangOpts().CPlusPlus11)
7715 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7716 if (!
Ctx.getLangOpts().CPlusPlus23)
7717 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7718 return this->emitGetRefGlobal(*GlobalIndex, E);
7721 return this->emitGetPtrGlobal(*GlobalIndex, E);
7725 auto revisit = [&](
const VarDecl *VD,
7726 bool IsConstexprUnknown =
true) ->
bool {
7728 IsConstexprUnknown);
7733 if (!this->emitPopCC(E))
7736 if (VarState.notCreated())
7744 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
7746 if (
auto It = this->LambdaCaptures.find(D);
7747 It != this->LambdaCaptures.end()) {
7748 auto [Offset, IsPtr] = It->second;
7751 return this->emitGetThisFieldPtr(Offset, E);
7752 return this->emitGetPtrThisField(Offset, E);
7756 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7757 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7758 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7762 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7763 return this->
visit(BD->getBinding());
7767 return this->emitDummyPtr(D, E);
7771 const auto *VD = dyn_cast<VarDecl>(D);
7773 return this->emitError(E);
7776 if (!
Ctx.getLangOpts().CPlusPlus) {
7780 return revisit(VD,
false);
7781 return this->emitDummyPtr(D, E);
7785 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7786 if (T.isConstant(
Ctx.getASTContext()))
7788 return T->isReferenceType();
7792 typeShouldBeVisited(DeclType)) {
7794 Init && !
Init->isValueDependent()) {
7800 (void)
Init->EvaluateAsInitializer(
V,
Ctx.getASTContext(), VD, Notes,
7814 bool IsConstexprUnknown = !DeclType.
isConstant(
Ctx.getASTContext()) &&
7819 return revisit(VD, IsConstexprUnknown);
7820 }
else if (
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7821 return revisit(VD,
true);
7828 return this->emitDummyPtr(
7832template <
class Emitter>
7838template <
class Emitter>
7848 if (!
C->destroyLocals())
7854template <
class Emitter>
7855unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7858 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7860 return Ty->getAsCXXRecordDecl();
7862 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7863 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7865 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7869template <
class Emitter>
7876 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7881 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7882 getFPOptions(E), E);
7884 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7885 getFPOptions(E), E);
7889 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7894 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7896 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7900 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7904 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7905 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7912template <
class Emitter>
7915 assert(FromT != ToT);
7918 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7920 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7922 return this->emitCast(FromT, ToT, E);
7926template <
class Emitter>
7927bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7931 return this->
discard(SubExpr);
7933 if (!this->visit(SubExpr))
7936 if (!this->emitConstUint8(0, SubExpr))
7938 return this->emitArrayElemPtrPopUint8(SubExpr);
7942 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7946template <
class Emitter>
7947bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7948 assert(!DiscardResult);
7952 if (!this->emitArrayElem(ElemT, 0, E))
7955 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7958 if (!this->emitCast(ElemT,
PT_Bool, E))
7963 LabelTy LabelTrue = this->getLabel();
7964 if (!this->jumpTrue(LabelTrue, E))
7967 if (!this->emitArrayElemPop(ElemT, 1, E))
7970 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7973 if (!this->emitCast(ElemT,
PT_Bool, E))
7977 LabelTy EndLabel = this->getLabel();
7978 this->jump(EndLabel, E);
7980 this->emitLabel(LabelTrue);
7981 if (!this->emitPopPtr(E))
7983 if (!this->emitConstBool(
true, E))
7986 this->fallthrough(EndLabel);
7987 this->emitLabel(EndLabel);
7992template <
class Emitter>
7993bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
8004 LHSIsComplex =
true;
8005 ElemT = classifyComplexElementType(LHS->
getType());
8006 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
8007 if (!this->visit(LHS))
8009 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
8012 LHSIsComplex =
false;
8014 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
8015 if (!this->visit(LHS))
8017 if (!this->emitSetLocal(LHST, LHSOffset, E))
8024 RHSIsComplex =
true;
8025 ElemT = classifyComplexElementType(RHS->
getType());
8026 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
8027 if (!this->visit(RHS))
8029 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
8032 RHSIsComplex =
false;
8034 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
8035 if (!this->visit(RHS))
8037 if (!this->emitSetLocal(RHST, RHSOffset, E))
8041 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
8042 bool IsComplex) ->
bool {
8044 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
8046 return this->emitArrayElemPop(ElemT, Index, E);
8048 return this->emitGetLocal(ElemT, LocalOffset, E);
8051 for (
unsigned I = 0; I != 2; ++I) {
8053 if (!getElem(LHSOffset, I, LHSIsComplex))
8055 if (!getElem(RHSOffset, I, RHSIsComplex))
8058 if (!this->emitEQ(ElemT, E))
8061 if (!this->emitCastBoolUint8(E))
8066 if (!this->emitAddUint8(E))
8068 if (!this->emitConstUint8(2, E))
8072 if (!this->emitEQUint8(E))
8075 if (!this->emitNEUint8(E))
8082 return this->emitCast(
PT_Bool, ResT, E);
8089template <
class Emitter>
8090bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
8093 assert(!
R->hasTrivialDtor());
8096 const Function *DtorFunc = getFunction(Dtor);
8099 assert(DtorFunc->hasThisPointer());
8100 assert(DtorFunc->getNumParams() == 1);
8101 return this->emitCall(DtorFunc, 0, Loc);
8106template <
class Emitter>
8107bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
8119 return this->emitPopPtr(Loc);
8121 for (ssize_t I = N - 1; I >= 1; --I) {
8122 if (!this->emitConstUint64(I, Loc))
8124 if (!this->emitArrayElemPtrUint64(Loc))
8126 if (!this->emitDestructionPop(ElemDesc, Loc))
8130 if (!this->emitConstUint64(0, Loc))
8132 if (!this->emitArrayElemPtrPopUint64(Loc))
8134 return this->emitDestructionPop(ElemDesc, Loc);
8139 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
8144template <
class Emitter>
8145bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E,
bool CU) {
8146 assert(!DiscardResult &&
"Should've been checked before");
8147 unsigned DummyID = P.getOrCreateDummy(D, CU);
8149 if (!this->emitGetPtrGlobal(DummyID, E))
8157 return this->emitDecayPtr(
PT_Ptr, PT, E);
8163template <
class Emitter>
8164bool Compiler<Emitter>::emitFloat(
const APFloat &F,
const Expr *E) {
8166 return this->emitConstFloat(
Floating(F), E);
8168 APInt I = F.bitcastToAPInt();
8169 return this->emitConstFloat(
8170 Floating(
const_cast<uint64_t *
>(I.getRawData()),
8171 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
8182template <
class Emitter>
8183bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
8196 if (!this->emitGetPtrLocal(*LocalIndex, E))
8206 if (!this->visit(SubExpr))
8208 }
else if (
OptPrimType FromT = classify(SubExpr)) {
8209 unsigned TempOffset =
8210 allocateLocalPrimitive(SubExpr, *FromT,
true);
8211 if (!this->visit(SubExpr))
8213 if (!this->emitSetLocal(*FromT, TempOffset, E))
8215 if (!this->emitGetPtrLocal(TempOffset, E))
8222 if (!this->emitBitCast(E))
8224 return DiscardResult ? this->emitPopPtr(E) :
true;
8228 const llvm::fltSemantics *TargetSemantics =
nullptr;
8230 TargetSemantics = &Ctx.getFloatSemantics(ToType);
8236 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
8238 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
8239 ResultBitWidth, TargetSemantics,
8244 return this->emitPop(*ToT, E);
8253template <
class Emitter>
8254bool Compiler<Emitter>::emitHLSLAggregateSplat(
PrimType SrcT,
8259 unsigned NumElems = 0;
8262 NumElems = VT->getNumElements();
8263 ElemType = VT->getElementType();
8265 NumElems = MT->getNumElementsFlattened();
8266 ElemType = MT->getElementType();
8269 PrimType ElemT = classifyPrim(ElemType);
8270 for (
unsigned I = 0; I != NumElems; ++I) {
8271 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8273 if (!this->emitPrimCast(SrcT, ElemT, ElemType, E))
8275 if (!this->emitInitElem(ElemT, I, E))
8285 QualType ArrElemType = CAT->getElementType();
8286 unsigned ArrSize = CAT->getZExtSize();
8289 for (
unsigned I = 0; I != ArrSize; ++I) {
8290 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8292 if (!this->emitPrimCast(SrcT, *ElemT, ArrElemType, E))
8294 if (!this->emitInitElem(*ElemT, I, E))
8298 for (
unsigned I = 0; I != ArrSize; ++I) {
8299 if (!this->emitConstUint32(I, E))
8301 if (!this->emitArrayElemPtrUint32(E))
8303 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, ArrElemType, E))
8305 if (!this->emitFinishInitPop(E))
8315 const Record *
R = getRecord(DestType);
8319 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8321 const Record::Base *B =
R->getBase(BS.getType());
8323 if (!this->emitGetPtrBase(B->Offset, E))
8325 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, BS.getType(), E))
8327 if (!this->emitFinishInitPop(E))
8332 for (
const Record::Field &F :
R->fields()) {
8333 if (F.isUnnamedBitField())
8336 QualType FieldType = F.Decl->getType();
8338 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8340 if (!this->emitPrimCast(SrcT, *FieldT, FieldType, E))
8342 if (F.isBitField()) {
8343 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8346 if (!this->emitInitField(*FieldT, F.Offset, E))
8350 if (!this->emitGetPtrField(F.Offset, E))
8352 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, FieldType, E))
8354 if (!this->emitPopPtr(E))
8367template <
class Emitter>
8368unsigned Compiler<Emitter>::countHLSLFlatElements(
QualType Ty) {
8371 return VT->getNumElements();
8373 return MT->getNumElementsFlattened();
8377 return CAT->getZExtSize() * countHLSLFlatElements(CAT->getElementType());
8381 const Record *
R = getRecord(Ty);
8385 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8387 Count += countHLSLFlatElements(BS.getType());
8389 for (
const Record::Field &F :
R->fields()) {
8390 if (F.isUnnamedBitField())
8392 Count += countHLSLFlatElements(F.Decl->getType());
8397 if (canClassify(Ty))
8406template <
class Emitter>
8407bool Compiler<Emitter>::emitHLSLFlattenAggregate(
8408 QualType SrcType,
unsigned SrcOffset,
8413 auto saveToLocal = [&](
PrimType T) ->
bool {
8414 unsigned Offset = allocateLocalPrimitive(E, T,
true);
8415 if (!this->emitSetLocal(T, Offset, E))
8417 Elements.push_back({Offset, T});
8423 unsigned Offset = allocateLocalPrimitive(E,
PT_Ptr,
true);
8424 if (!this->emitSetLocal(
PT_Ptr, Offset, E))
8425 return std::nullopt;
8430 unsigned NumElems = 0;
8433 NumElems = VT->getNumElements();
8434 ElemType = VT->getElementType();
8436 NumElems = MT->getNumElementsFlattened();
8437 ElemType = MT->getElementType();
8440 PrimType ElemT = classifyPrim(ElemType);
8441 for (
unsigned I = 0; I != NumElems && Elements.size() < MaxElements; ++I) {
8442 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8444 if (!this->emitArrayElemPop(ElemT, I, E))
8446 if (!saveToLocal(ElemT))
8456 QualType ArrElemType = CAT->getElementType();
8457 unsigned ArrSize = CAT->getZExtSize();
8460 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8461 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8463 if (!this->emitArrayElemPop(*ElemT, I, E))
8465 if (!saveToLocal(*ElemT))
8469 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8470 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8472 if (!this->emitConstUint32(I, E))
8474 if (!this->emitArrayElemPtrPopUint32(E))
8479 if (!emitHLSLFlattenAggregate(ArrElemType, *ElemPtrOffset, Elements,
8490 const Record *
R = getRecord(SrcType);
8494 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8496 if (Elements.size() >= MaxElements)
8498 const Record::Base *B =
R->getBase(BS.getType());
8500 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8502 if (!this->emitGetPtrBasePop(B->Offset,
false, E))
8507 if (!emitHLSLFlattenAggregate(BS.getType(), *BasePtrOffset, Elements,
8513 for (
const Record::Field &F :
R->fields()) {
8514 if (Elements.size() >= MaxElements)
8516 if (F.isUnnamedBitField())
8519 QualType FieldType = F.Decl->getType();
8520 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8522 if (!this->emitGetPtrFieldPop(F.Offset, E))
8526 if (!this->emitLoadPop(*FieldT, E))
8528 if (!saveToLocal(*FieldT))
8532 if (!FieldPtrOffset)
8534 if (!emitHLSLFlattenAggregate(FieldType, *FieldPtrOffset, Elements,
8550template <
class Emitter>
8551bool Compiler<Emitter>::emitHLSLConstructAggregate(
8557 const auto &Src = Elements[ElemIdx++];
8558 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
8560 return this->emitPrimCast(Src.Type, DestT, DestQT, E);
8564 unsigned NumElems = 0;
8567 NumElems = VT->getNumElements();
8568 ElemType = VT->getElementType();
8570 NumElems = MT->getNumElementsFlattened();
8571 ElemType = MT->getElementType();
8574 PrimType DestElemT = classifyPrim(ElemType);
8575 for (
unsigned I = 0; I != NumElems; ++I) {
8576 if (!loadAndCast(DestElemT, ElemType))
8578 if (!this->emitInitElem(DestElemT, I, E))
8588 QualType ArrElemType = CAT->getElementType();
8589 unsigned ArrSize = CAT->getZExtSize();
8592 for (
unsigned I = 0; I != ArrSize; ++I) {
8593 if (!loadAndCast(*ElemT, ArrElemType))
8595 if (!this->emitInitElem(*ElemT, I, E))
8599 for (
unsigned I = 0; I != ArrSize; ++I) {
8600 if (!this->emitConstUint32(I, E))
8602 if (!this->emitArrayElemPtrUint32(E))
8604 if (!emitHLSLConstructAggregate(ArrElemType, Elements, ElemIdx, E))
8606 if (!this->emitFinishInitPop(E))
8616 const Record *
R = getRecord(DestType);
8620 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8622 const Record::Base *B =
R->getBase(BS.getType());
8624 if (!this->emitGetPtrBase(B->Offset, E))
8626 if (!emitHLSLConstructAggregate(BS.getType(), Elements, ElemIdx, E))
8628 if (!this->emitFinishInitPop(E))
8633 for (
const Record::Field &F :
R->fields()) {
8634 if (F.isUnnamedBitField())
8637 QualType FieldType = F.Decl->getType();
8639 if (!loadAndCast(*FieldT, FieldType))
8641 if (F.isBitField()) {
8642 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8645 if (!this->emitInitField(*FieldT, F.Offset, E))
8649 if (!this->emitGetPtrField(F.Offset, E))
8651 if (!emitHLSLConstructAggregate(FieldType, Elements, ElemIdx, E))
8653 if (!this->emitPopPtr(E))
static void emit(Program &P, llvm::SmallVectorImpl< std::byte > &Code, const T &Val, bool &Success)
Helper to write bytecode and bail out if 32-bit offsets become invalid.
static void emitCleanup(CIRGenFunction &cgf, cir::CleanupScopeOp cleanupScope, EHScopeStack::Cleanup *cleanup, EHScopeStack::Cleanup::Flags flags, Address activeFlag)
static uint32_t getBitWidth(const Expr *E)
#define EMIT_ARITH_OP(OP)
static CharUnits AlignOfType(QualType T, const ASTContext &ASTCtx, UnaryExprOrTypeTrait Kind)
static const Expr * stripDerivedToBaseCasts(const Expr *E)
static const Expr * stripCheckedDerivedToBaseCasts(const Expr *E)
static bool hasTrivialDefaultCtorParent(const FieldDecl *FD)
static bool initNeedsOverridenLoc(const CXXCtorInitializer *Init)
Result
Implement __builtin_bit_cast and related operations.
a trap message and trap category.
llvm::APInt getValue() const
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
const LValueBase getLValueBase() const
APValue & getArrayInitializedElt(unsigned I)
ArrayRef< LValuePathEntry > getLValuePath() const
APValue & getStructField(unsigned i)
const FieldDecl * getUnionField() const
unsigned getStructNumFields() const
bool isMemberPointerToDerivedMember() const
unsigned getArrayInitializedElts() const
unsigned getStructNumBases() const
const ValueDecl * getMemberPointerDecl() const
APValue & getUnionValue()
APValue & getArrayFiller()
ArrayRef< const CXXRecordDecl * > getMemberPointerPath() const
bool isMemberPointer() const
unsigned getArraySize() const
@ None
There is no such object (it's outside its lifetime).
bool isNullPointer() const
APValue & getStructBase(unsigned i)
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
unsigned getPreferredTypeAlign(QualType T) const
Return the "preferred" alignment of the specified type T for the current target, in bits.
const LangOptions & getLangOpts() const
unsigned getOpenMPDefaultSimdAlign(QualType T) const
Get default simd alignment of the specified complete type in bits.
TypeInfoChars getTypeInfoDataSizeInChars(QualType T) const
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
llvm::FixedPointSemantics getFixedPointSemantics(QualType Ty) const
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
const VariableArrayType * getAsVariableArrayType(QualType T) const
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
AbstractConditionalOperator - An abstract base class for ConditionalOperator and BinaryConditionalOpe...
Expr * getCond() const
getCond - Return the expression representing the condition for the ?
Expr * getTrueExpr() const
getTrueExpr - Return the subexpression representing the value of the expression if the condition eval...
Expr * getFalseExpr() const
getFalseExpr - Return the subexpression representing the value of the expression if the condition eva...
AddrLabelExpr - The GNU address of label extension, representing &&label.
Represents the index of the current element of an array being initialized by an ArrayInitLoopExpr.
Represents a loop initializing the elements of an array.
llvm::APInt getArraySize() const
OpaqueValueExpr * getCommonExpr() const
Get the common subexpression shared by all initializations (the source array).
Expr * getSubExpr() const
Get the initializer to use for each array element.
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
Expr * getLHS()
An array access can be written A[4] or 4[A] (both are equivalent).
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
uint64_t getValue() const
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Attr - This represents one attribute.
Represents an attribute applied to a statement.
ArrayRef< const Attr * > getAttrs() const
Represents a C++ declaration that introduces decls from somewhere else.
A builtin binary operation expression such as "x + y" or "x <= y".
static bool isLogicalOp(Opcode Opc)
static bool isComparisonOp(Opcode Opc)
static bool isShiftOp(Opcode Opc)
static bool isCommaOp(Opcode Opc)
static Opcode getOpForCompoundAssignment(Opcode Opc)
static bool isPtrMemOp(Opcode Opc)
predicates to categorize the respective opcodes.
static bool isAssignmentOp(Opcode Opc)
static bool isCompoundAssignmentOp(Opcode Opc)
static bool isBitwiseOp(Opcode Opc)
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
BreakStmt - This represents a break.
Represents a C++2a __builtin_bit_cast(T, v) expression.
Represents a base class of a C++ class.
Represents binding an expression to a temporary.
const Expr * getSubExpr() const
A boolean literal, per ([C++ lex.bool] Boolean literals).
Represents a call to a C++ constructor.
bool isElidable() const
Whether this construction is elidable.
Expr * getArg(unsigned Arg)
Return the specified argument.
bool requiresZeroInitialization() const
Whether this construction first requires zero-initialization before the initializer is called.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will (ultimately) call.
unsigned getNumArgs() const
Return the number of arguments to the constructor call.
Represents a C++ constructor within a class.
bool isCopyOrMoveConstructor(unsigned &TypeQuals) const
Determine whether this is a copy or move constructor.
Represents a C++ base or member initializer.
A default argument (C++ [dcl.fct.default]).
A use of a default initializer in a constructor or in aggregate initialization.
Expr * getExpr()
Get the initialization expression that will be used.
Represents a delete expression for memory deallocation and destructor calls, e.g.
FunctionDecl * getOperatorDelete() const
bool isGlobalDelete() const
Represents a C++ destructor within a class.
A C++ dynamic_cast expression (C++ [expr.dynamic.cast]).
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
DeclStmt * getBeginStmt()
DeclStmt * getLoopVarStmt()
DeclStmt * getRangeStmt()
Represents a call to an inherited base class constructor from an inheriting constructor.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will call.
Represents a static or instance method of a struct/union/class.
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
bool isMoveAssignmentOperator() const
Determine whether this is a move assignment operator.
bool isCopyAssignmentOperator() const
Determine whether this is a copy-assignment operator, regardless of whether it was declared implicitl...
bool isLambdaStaticInvoker() const
Determine whether this is a lambda closure type's static member function that is used for the result ...
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
QualType getAllocatedType() const
std::optional< Expr * > getArraySize()
This might return std::nullopt even if isArray() returns true, since there might not be an array size...
Expr * getPlacementArg(unsigned I)
unsigned getNumPlacementArgs() const
FunctionDecl * getOperatorNew() const
Expr * getInitializer()
The initializer of this new-expression.
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
The null pointer literal (C++11 [lex.nullptr])
Represents a list-initialization with parenthesis.
MutableArrayRef< Expr * > getInitExprs()
Represents a C++ struct/union/class.
bool hasTrivialDefaultConstructor() const
Determine whether this class has a trivial default constructor (C++11 [class.ctor]p5).
bool isGenericLambda() const
Determine whether this class describes a generic lambda function object (i.e.
capture_const_range captures() const
CXXMethodDecl * getLambdaCallOperator() const
Retrieve the lambda call operator of the closure type if this is a closure type.
A C++ reinterpret_cast expression (C++ [expr.reinterpret.cast]).
A rewritten comparison expression that was originally written using operator syntax.
Expr * getSemanticForm()
Get an equivalent semantic form for this expression.
An expression "T()" which creates an rvalue of a non-class type T.
Implicit construction of a std::initializer_list<T> object from an array temporary within list-initia...
Represents the this expression in C++.
A C++ throw-expression (C++ [except.throw]).
const Expr * getSubExpr() const
CXXTryStmt - A C++ try block, including all handlers.
CompoundStmt * getTryBlock()
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
bool isTypeOperand() const
QualType getTypeOperand(const ASTContext &Context) const
Retrieves the type operand of this typeid() expression after various required adjustments (removing r...
Expr * getExprOperand() const
bool isPotentiallyEvaluated() const
Determine whether this typeid has a type operand which is potentially evaluated, per C++11 [expr....
A Microsoft C++ __uuidof expression, which gets the _GUID that corresponds to the supplied type or ex...
MSGuidDecl * getGuidDecl() const
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
FunctionDecl * getDirectCallee()
If the callee is a FunctionDecl, return it. Otherwise return null.
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this call.
Expr ** getArgs()
Retrieve the call arguments.
QualType getCallReturnType(const ASTContext &Ctx) const
getCallReturnType - Get the return type of the call expr.
CaseStmt - Represent a case statement.
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
path_iterator path_begin()
CastKind getCastKind() const
llvm::iterator_range< path_iterator > path()
Path through the class hierarchy taken by casts between base and derived classes (see implementation ...
const FieldDecl * getTargetUnionField() const
CharUnits - This is an opaque type for sizes expressed in character units.
bool isZero() const
isZero - Test whether the quantity equals zero.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
unsigned getValue() const
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Expr * getChosenSubExpr() const
getChosenSubExpr - Return the subexpression chosen according to the condition.
Complex values, per C99 6.2.5p11.
QualType getElementType() const
CompoundAssignOperator - For compound assignments (e.g.
QualType getComputationLHSType() const
QualType getComputationResultType() const
CompoundLiteralExpr - [C99 6.5.2.5].
const Expr * getInitializer() const
CompoundStmt - This represents a group of statements like { stmt stmt }.
Represents the specialization of a concept - evaluates to a prvalue of type bool.
bool isSatisfied() const
Whether or not the concept with the given arguments was satisfied when the expression was created.
Represents the canonical version of C arrays with a specified constant size.
uint64_t getZExtSize() const
Return the size zero-extended as a uint64_t.
ConstantExpr - An expression that occurs in a constant context and optionally the result of evaluatin...
APValue getAPValueResult() const
bool hasAPValueResult() const
Represents a concrete matrix type with constant number of rows and columns.
ContinueStmt - This represents a continue.
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
Expr * getSrcExpr() const
getSrcExpr - Return the Expr to be converted.
DeclContext * getParent()
getParent - Returns the containing DeclContext.
A reference to a declared variable, function, enum, etc.
DeclStmt - Adaptor class for mixing declarations with statements and expressions.
Decl - This represents one declaration (or definition), e.g.
bool isInvalidDecl() const
OverloadedOperatorKind getCXXOverloadedOperator() const
If this name is the name of an overloadable operator in C++ (e.g., operator+), retrieve the kind of o...
bool isAnyOperatorNew() const
InitListExpr * getUpdater() const
DoStmt - This represents a 'do/while' stmt.
const Expr * getBase() const
Represents a reference to emded data.
ChildElementIter< false > begin()
Represents an expression – generally a full-expression – that introduces cleanups to be run at the en...
This represents one expression.
const Expr * skipRValueSubobjectAdjustments(SmallVectorImpl< const Expr * > &CommaLHS, SmallVectorImpl< SubobjectAdjustment > &Adjustments) const
Walk outwards from an expression we want to bind a reference to and find the expression whose lifetim...
bool isValueDependent() const
Determines whether the value of this expression depends on.
Expr * IgnoreImplicit() LLVM_READONLY
Skip past any implicit AST nodes which might surround this expression until reaching a fixed point.
bool containsErrors() const
Whether this expression contains subexpressions which had errors.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
bool isLValue() const
isLValue - True if this expression is an "l-value" according to the rules of the current language.
bool HasSideEffects(const ASTContext &Ctx, bool IncludePossibleEffects=true) const
HasSideEffects - This routine returns true for all those expressions which have any effect other than...
bool isTemporaryObject(ASTContext &Ctx, const CXXRecordDecl *TempTy) const
Determine whether the result of this expression is a temporary object of the given class type.
bool refersToBitField() const
Returns true if this expression is a gl-value that potentially refers to a bit-field.
An expression trait intrinsic.
ExtVectorElementExpr - This represents access to specific elements of a vector, and may occur on the ...
void getEncodedElementAccess(SmallVectorImpl< uint32_t > &Elts) const
getEncodedElementAccess - Encode the elements accessed into an llvm aggregate Constant of ConstantInt...
Represents a member of a struct/union/class.
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
llvm::APInt getValue() const
Returns an internal integer representation of the literal.
llvm::APFloat getValue() const
ForStmt - This represents a 'for (init;cond;inc)' stmt.
VarDecl * getConditionVariable() const
Retrieve the variable declared in this "for" statement, if any.
DeclStmt * getConditionVariableDeclStmt()
If this ForStmt has a condition variable, return the faux DeclStmt associated with the creation of th...
const Expr * getSubExpr() const
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
bool isFunctionTemplateSpecialization() const
Determine whether this function is a function template specialization.
FunctionTemplateDecl * getDescribedFunctionTemplate() const
Retrieves the function template that is described by this function declaration.
unsigned getBuiltinID(bool ConsiderWrapperFunctions=false) const
Returns a value indicating whether this function corresponds to a builtin function.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
const TemplateArgumentList * getTemplateSpecializationArgs() const
Retrieve the template arguments used to produce this function template specialization from the primar...
bool isUsableAsGlobalAllocationFunctionInConstantEvaluation(UnsignedOrNone *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions described in i...
bool isDefaulted() const
Whether this function is defaulted.
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Declaration of a template function.
FunctionDecl * findSpecialization(ArrayRef< TemplateArgument > Args, void *&InsertPos)
Return the specialization with the provided arguments if it exists, otherwise return the insertion po...
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
Expr * getResultExpr()
Return the result expression of this controlling expression.
IfStmt - This represents an if/then/else.
bool isNonNegatedConsteval() const
bool isNegatedConsteval() const
DeclStmt * getConditionVariableDeclStmt()
If this IfStmt has a condition variable, return the faux DeclStmt associated with the creation of tha...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "if" statement, if any.
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
const Expr * getSubExpr() const
Represents an implicitly-generated value initialization of an object of a given type.
Represents a field injected from an anonymous union/struct into the parent scope.
Describes an C or C++ initializer list.
Expr * getArrayFiller()
If this initializer list initializes an array with more elements than there are initializers in the l...
ArrayRef< Expr * > inits() const
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
capture_init_iterator capture_init_begin()
Retrieve the first initialization argument for this lambda expression (which initializes the first ca...
CXXRecordDecl * getLambdaClass() const
Retrieve the class that corresponds to the lambda.
bool isCompatibleWith(ClangABI Version) const
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
const Stmt * getNamedLoopOrSwitch() const
If this is a named break/continue, get the loop or switch statement that this targets.
APValue & getAsAPValue() const
Get the value of this MSGuidDecl as an APValue.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
StorageDuration getStorageDuration() const
Retrieve the storage duration for the materialized temporary.
Expr * getSubExpr() const
Retrieve the temporary-generating subexpression whose value will be materialized into a glvalue.
ValueDecl * getExtendingDecl()
Get the declaration which triggered the lifetime-extension of this temporary, if any.
LifetimeExtendedTemporaryDecl * getLifetimeExtendedTemporaryDecl()
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
ValueDecl * getMemberDecl() const
Retrieve the member declaration to which this expression refers.
A pointer to member type per C++ 8.3.3 - Pointers to members.
This represents a decl that may have a name.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
Represents a C++ namespace alias.
ObjCArrayLiteral - used for objective-c array containers; as in: @["Hello", NSApp,...
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
ObjCDictionaryLiteral - AST node to represent objective-c dictionary literals; as in:"name" : NSUserN...
ObjCEncodeExpr, used for @encode in Objective-C.
QualType getEncodedType() const
SourceLocation getAtLoc() const
bool isExpressibleAsConstantInitializer() const
ObjCStringLiteral, used for Objective-C string literals i.e.
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Expr * getIndexExpr(unsigned Idx)
const OffsetOfNode & getComponent(unsigned Idx) const
unsigned getNumComponents() const
Helper class for OffsetOfExpr.
unsigned getArrayExprIndex() const
For an array element node, returns the index into the array of expressions.
@ Array
An index into an array.
Kind getKind() const
Determine what kind of offsetof node this is.
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
Expr * getSourceExpr() const
The source expression of an opaque value expression is the expression which originally generated the ...
Expr * getSelectedExpr() const
ParenExpr - This represents a parenthesized expression, e.g.
const Expr * getSubExpr() const
Represents a parameter to a function.
PointerType - C99 6.7.5.1 - Pointer Declarators.
QualType getPointeeType() const
[C99 6.4.2.2] - A predefined identifier such as func.
StringLiteral * getFunctionName()
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
Expr * getResultExpr()
Return the result-bearing expression, or null if there is none.
ArrayRef< Expr * > semantics()
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
QualType withConst() const
bool isNull() const
Return true if this QualType doesn't point to a type yet.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
bool isConstant(const ASTContext &Ctx) const
bool isConstQualified() const
Determine whether this type is const-qualified.
Represents a struct/union/class.
Frontend produces RecoveryExprs on semantic errors that prevent creating other well-formed expression...
Base for LValueReferenceType and RValueReferenceType.
C++2a [expr.prim.req]: A requires-expression provides a concise way to express requirements on templa...
bool isSatisfied() const
Whether or not the requires clause is satisfied.
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
SourceLocation getLocation() const
std::string ComputeName(ASTContext &Context) const
Scope - A scope is a transient data structure that is used while parsing the program.
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
llvm::APSInt getShuffleMaskIdx(unsigned N) const
unsigned getNumSubExprs() const
getNumSubExprs - Return the size of the SubExprs array.
Expr * getExpr(unsigned Index)
getExpr - Return the Expr at the specified index.
Represents an expression that computes the length of a parameter pack.
unsigned getPackLength() const
Retrieve the length of the parameter pack.
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
APValue EvaluateInContext(const ASTContext &Ctx, const Expr *DefaultExpr) const
Return the result of evaluating this SourceLocExpr in the specified (and possibly null) default argum...
Represents a C++11 static_assert declaration.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
CompoundStmt * getSubStmt()
bool Visit(PTR(Stmt) S, ParamTys... P)
Stmt - This represents one statement.
StmtClass getStmtClass() const
StringLiteral - This represents a string literal expression, e.g.
unsigned getLength() const
static StringLiteral * Create(const ASTContext &Ctx, StringRef Str, StringLiteralKind Kind, bool Pascal, QualType Ty, ArrayRef< SourceLocation > Locs)
This is the "fully general" constructor that allows representation of strings formed from one or more...
uint32_t getCodeUnit(size_t i) const
unsigned getCharByteWidth() const
Represents a reference to a non-type template parameter that has been substituted with a template arg...
Expr * getReplacement() const
const SwitchCase * getNextSwitchCase() const
SwitchStmt - This represents a 'switch' stmt.
VarDecl * getConditionVariable()
Retrieve the variable declared in this "switch" statement, if any.
SwitchCase * getSwitchCaseList()
DeclStmt * getConditionVariableDeclStmt()
If this SwitchStmt has a condition variable, return the faux DeclStmt associated with the creation of...
Represents the declaration of a struct/union/class/enum.
bool isCompleteDefinition() const
Return true if this decl has its body fully specified.
A template argument list.
ArrayRef< TemplateArgument > asArray() const
Produce this as an array ref.
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
bool getBoolValue() const
const APValue & getAPValue() const
bool isStoredAsBoolean() const
The base class of the type hierarchy.
bool isBooleanType() const
bool isLiteralType(const ASTContext &Ctx) const
Return true if this is a literal type (C++11 [basic.types]p10)
bool isIncompleteArrayType() const
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isVoidPointerType() const
bool isConstantSizeType() const
Return true if this is not a variable sized type, according to the rules of C99 6....
bool isFunctionPointerType() const
bool isConstantMatrixType() const
bool isPointerType() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
bool isEnumeralType() const
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isIntegralOrEnumerationType() const
Determine whether this type is an integral or enumeration type.
bool isSpecificBuiltinType(unsigned K) const
Test for a particular builtin type.
bool isDependentType() const
Whether this type is a dependent type, meaning that its definition somehow depends on a template para...
bool isAnyComplexType() const
bool isFixedPointType() const
Return true if this is a fixed point type according to ISO/IEC JTC1 SC22 WG14 N1169.
const Type * getBaseElementTypeUnsafe() const
Get the base element type of this type, potentially discarding type qualifiers.
bool isMemberPointerType() const
bool isAtomicType() const
EnumDecl * castAsEnumDecl() const
bool isStdByteType() const
const ArrayType * getAsArrayTypeUnsafe() const
A variant of getAs<> for array types which silently discards qualifiers from the outermost type.
bool isPointerOrReferenceType() const
bool isFunctionType() const
bool isVectorType() const
bool isRealFloatingType() const
Floating point categories.
const T * getAsCanonical() const
If this type is canonically the specified type, return its canonical type cast to that specified type...
bool isFloatingType() const
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
bool isSizelessVectorType() const
Returns true for all scalable vector types.
Base class for declarations which introduce a typedef-name.
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
QualType getArgumentType() const
QualType getTypeOfArgument() const
Gets the argument type, or the type of the argument expression, whichever is appropriate.
bool isArgumentType() const
UnaryExprOrTypeTrait getKind() const
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Expr * getSubExpr() const
bool canOverflow() const
Returns true if the unary operator can cause an overflow.
Represents C++ using-directive.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
bool isWeak() const
Determine whether this symbol is weakly-imported, or declared with the weak or weak-ref attr.
Represents a variable declaration or definition.
bool isConstexpr() const
Whether this variable is (C++11) constexpr.
bool isInitCapture() const
Whether this variable is the implicit variable for a lambda init-capture.
APValue * evaluateValue() const
Attempt to evaluate the value of the initializer attached to this declaration, and produce notes expl...
bool isStaticDataMember() const
Determines whether this is a static data member.
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
bool hasConstantInitialization() const
Determine whether this variable has constant initialization.
bool isStaticLocal() const
Returns true if a variable with function scope is a static local variable.
const Expr * getInit() const
bool isLocalVarDecl() const
Returns true for local variable declarations other than parameters.
const Expr * getAnyInitializer() const
Get the initializer for this variable, no matter which declaration it is attached to.
Represents a GCC generic vector type.
unsigned getNumElements() const
QualType getElementType() const
WhileStmt - This represents a 'while' stmt.
DeclStmt * getConditionVariableDeclStmt()
If this WhileStmt has a condition variable, return the faux DeclStmt associated with the creation of ...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "while" statement, if any.
A memory block, either on the stack or in the heap.
void invokeDtor()
Invokes the Destructor.
Compilation context for expressions.
llvm::SmallVector< InitLink > InitStack
bool VisitArrayInitIndexExpr(const ArrayInitIndexExpr *E)
bool VisitCXXDeleteExpr(const CXXDeleteExpr *E)
bool VisitOffsetOfExpr(const OffsetOfExpr *E)
bool visitContinueStmt(const ContinueStmt *S)
bool VisitCharacterLiteral(const CharacterLiteral *E)
bool visitArrayElemInit(unsigned ElemIndex, const Expr *Init, OptPrimType InitT)
Pointer to the array(not the element!) must be on the stack when calling this.
UnsignedOrNone allocateLocal(DeclTy &&Decl, QualType Ty=QualType(), ScopeKind=ScopeKind::Block)
Allocates a space storing a local given its type.
bool VisitCXXParenListInitExpr(const CXXParenListInitExpr *E)
bool VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E)
bool visitInitializerPop(const Expr *E)
Similar, but will also pop the pointer.
bool VisitCompoundLiteralExpr(const CompoundLiteralExpr *E)
bool visitBool(const Expr *E)
Visits an expression and converts it to a boolean.
bool VisitCXXDefaultInitExpr(const CXXDefaultInitExpr *E)
PrimType classifyPrim(QualType Ty) const
Classifies a known primitive type.
bool VisitTypeTraitExpr(const TypeTraitExpr *E)
bool VisitLambdaExpr(const LambdaExpr *E)
bool VisitMemberExpr(const MemberExpr *E)
llvm::DenseMap< const OpaqueValueExpr *, unsigned > OpaqueExprs
OpaqueValueExpr to location mapping.
bool VisitBinaryOperator(const BinaryOperator *E)
bool visitAttributedStmt(const AttributedStmt *S)
bool VisitPackIndexingExpr(const PackIndexingExpr *E)
bool visitAPValueInitializer(const APValue &Val, const Expr *E, QualType T)
bool VisitArraySubscriptExpr(const ArraySubscriptExpr *E)
bool VisitCallExpr(const CallExpr *E)
std::optional< uint64_t > ArrayIndex
Current argument index. Needed to emit ArrayInitIndexExpr.
bool VisitPseudoObjectExpr(const PseudoObjectExpr *E)
bool VisitCXXReinterpretCastExpr(const CXXReinterpretCastExpr *E)
const Function * getFunction(const FunctionDecl *FD)
Returns a function for the given FunctionDecl.
bool VisitFixedPointBinOp(const BinaryOperator *E)
bool VisitCastExpr(const CastExpr *E)
bool VisitObjCEncodeExpr(const ObjCEncodeExpr *E)
bool VisitFixedPointUnaryOperator(const UnaryOperator *E)
unsigned allocateLocalPrimitive(DeclTy &&Decl, PrimType Ty, bool IsConst, bool IsVolatile=false, ScopeKind SC=ScopeKind::Block)
Creates a local primitive value.
bool VisitComplexUnaryOperator(const UnaryOperator *E)
llvm::DenseMap< const SwitchCase *, LabelTy > CaseMap
bool VisitBlockExpr(const BlockExpr *E)
bool visitAPValue(const APValue &Val, PrimType ValType, const Expr *E)
Visit an APValue.
bool VisitCXXScalarValueInitExpr(const CXXScalarValueInitExpr *E)
bool VisitLogicalBinOp(const BinaryOperator *E)
bool visitCompoundStmt(const CompoundStmt *S)
Context & Ctx
Current compilation context.
bool visitDeclRef(const ValueDecl *D, const Expr *E)
Visit the given decl as if we have a reference to it.
bool visitBreakStmt(const BreakStmt *S)
bool visitExpr(const Expr *E, bool DestroyToplevelScope) override
bool visitForStmt(const ForStmt *S)
bool VisitDeclRefExpr(const DeclRefExpr *E)
bool VisitOpaqueValueExpr(const OpaqueValueExpr *E)
bool VisitArrayInitLoopExpr(const ArrayInitLoopExpr *E)
bool VisitStmtExpr(const StmtExpr *E)
bool VisitBuiltinBitCastExpr(const BuiltinBitCastExpr *E)
bool VisitFixedPointLiteral(const FixedPointLiteral *E)
const FunctionDecl * CompilingFunction
bool VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *E)
VarCreationState visitVarDecl(const VarDecl *VD, const Expr *Init, bool Toplevel=false)
Creates and initializes a variable from the given decl.
VariableScope< Emitter > * VarScope
Current scope.
bool visitDeclAndReturn(const VarDecl *VD, const Expr *Init, bool ConstantContext) override
Toplevel visitDeclAndReturn().
bool VisitCXXNewExpr(const CXXNewExpr *E)
const ValueDecl * InitializingDecl
bool VisitCompoundAssignOperator(const CompoundAssignOperator *E)
bool visit(const Expr *E) override
Evaluates an expression and places the result on the stack.
bool delegate(const Expr *E)
Just pass evaluation on to E.
bool visitLValueExpr(const Expr *E, bool DestroyToplevelScope) override
bool discard(const Expr *E)
Evaluates an expression for side effects and discards the result.
bool VisitCXXDefaultArgExpr(const CXXDefaultArgExpr *E)
CaseMap CaseLabels
Switch case mapping.
Record * getRecord(QualType Ty)
Returns a record from a record or pointer type.
const RecordType * getRecordTy(QualType Ty)
Returns a record type from a record or pointer type.
bool VisitCXXStdInitializerListExpr(const CXXStdInitializerListExpr *E)
bool visitInitList(ArrayRef< const Expr * > Inits, const Expr *ArrayFiller, const Expr *E)
bool VisitSizeOfPackExpr(const SizeOfPackExpr *E)
bool VisitPredefinedExpr(const PredefinedExpr *E)
bool VisitSourceLocExpr(const SourceLocExpr *E)
bool visitDeclStmt(const DeclStmt *DS, bool EvaluateConditionDecl=false)
bool emitCleanup()
Emits scope cleanup instructions.
bool VisitExtVectorElementExpr(const ExtVectorElementExpr *E)
bool VisitObjCStringLiteral(const ObjCStringLiteral *E)
bool VisitArrayTypeTraitExpr(const ArrayTypeTraitExpr *E)
bool visitInitializer(const Expr *E)
Compiles an initializer.
bool visitDtorCall(const VarDecl *VD, const APValue &Value) override
const Expr * SourceLocDefaultExpr
DefaultInit- or DefaultArgExpr, needed for SourceLocExpr.
bool VisitObjCArrayLiteral(const ObjCArrayLiteral *E)
UnsignedOrNone OptLabelTy
bool VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *E)
bool VisitPointerArithBinOp(const BinaryOperator *E)
Perform addition/subtraction of a pointer and an integer or subtraction of two pointers.
bool visitCallArgs(ArrayRef< const Expr * > Args, const FunctionDecl *FuncDecl, bool Activate, bool IsOperatorCall)
bool VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *E)
bool visitDefaultStmt(const DefaultStmt *S)
bool VisitObjCDictionaryLiteral(const ObjCDictionaryLiteral *E)
typename Emitter::LabelTy LabelTy
VarCreationState visitDecl(const VarDecl *VD)
bool VisitCXXDynamicCastExpr(const CXXDynamicCastExpr *E)
bool visitStmt(const Stmt *S)
bool VisitExpressionTraitExpr(const ExpressionTraitExpr *E)
bool VisitVectorUnaryOperator(const UnaryOperator *E)
bool VisitCXXConstructExpr(const CXXConstructExpr *E)
bool VisitCXXNullPtrLiteralExpr(const CXXNullPtrLiteralExpr *E)
bool VisitObjCBoxedExpr(const ObjCBoxedExpr *E)
bool VisitDesignatedInitUpdateExpr(const DesignatedInitUpdateExpr *E)
bool VisitCXXInheritedCtorInitExpr(const CXXInheritedCtorInitExpr *E)
bool VisitRecoveryExpr(const RecoveryExpr *E)
bool VisitRequiresExpr(const RequiresExpr *E)
bool Initializing
Flag inidicating if we're initializing an already created variable.
bool visitReturnStmt(const ReturnStmt *RS)
bool VisitCXXThrowExpr(const CXXThrowExpr *E)
bool VisitSubstNonTypeTemplateParmExpr(const SubstNonTypeTemplateParmExpr *E)
bool VisitChooseExpr(const ChooseExpr *E)
bool visitFunc(const FunctionDecl *F) override
bool visitCXXForRangeStmt(const CXXForRangeStmt *S)
bool visitCaseStmt(const CaseStmt *S)
bool VisitComplexBinOp(const BinaryOperator *E)
llvm::DenseMap< const ValueDecl *, Scope::Local > Locals
Variable to storage mapping.
bool VisitAbstractConditionalOperator(const AbstractConditionalOperator *E)
bool VisitCXXTypeidExpr(const CXXTypeidExpr *E)
UnsignedOrNone allocateTemporary(const Expr *E)
bool VisitBuiltinCallExpr(const CallExpr *E, unsigned BuiltinID)
bool VisitImplicitValueInitExpr(const ImplicitValueInitExpr *E)
bool VisitCXXRewrittenBinaryOperator(const CXXRewrittenBinaryOperator *E)
OptPrimType ReturnType
Type of the expression returned by the function.
bool VisitUnaryOperator(const UnaryOperator *E)
bool VisitFloatCompoundAssignOperator(const CompoundAssignOperator *E)
OptPrimType classify(const Expr *E) const
llvm::SmallVector< LabelInfo > LabelInfoStack
Stack of label information for loops and switch statements.
bool VisitGenericSelectionExpr(const GenericSelectionExpr *E)
bool visitDoStmt(const DoStmt *S)
bool VisitIntegerLiteral(const IntegerLiteral *E)
bool VisitInitListExpr(const InitListExpr *E)
bool VisitVectorBinOp(const BinaryOperator *E)
bool VisitStringLiteral(const StringLiteral *E)
bool VisitParenExpr(const ParenExpr *E)
bool VisitCXXNoexceptExpr(const CXXNoexceptExpr *E)
bool VisitShuffleVectorExpr(const ShuffleVectorExpr *E)
bool VisitPointerCompoundAssignOperator(const CompoundAssignOperator *E)
bool DiscardResult
Flag indicating if return value is to be discarded.
bool VisitEmbedExpr(const EmbedExpr *E)
bool VisitConvertVectorExpr(const ConvertVectorExpr *E)
bool VisitCXXThisExpr(const CXXThisExpr *E)
bool VisitConstantExpr(const ConstantExpr *E)
bool VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *E)
bool visitSwitchStmt(const SwitchStmt *S)
bool VisitCXXUuidofExpr(const CXXUuidofExpr *E)
bool VisitExprWithCleanups(const ExprWithCleanups *E)
bool visitAsLValue(const Expr *E)
bool visitWhileStmt(const WhileStmt *S)
bool visitIfStmt(const IfStmt *IS)
bool VisitAddrLabelExpr(const AddrLabelExpr *E)
bool canClassify(const Expr *E) const
bool VisitFloatingLiteral(const FloatingLiteral *E)
Program & P
Program to link to.
bool VisitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *E)
bool VariablesAreConstexprUnknown
bool VisitGNUNullExpr(const GNUNullExpr *E)
bool VisitImaginaryLiteral(const ImaginaryLiteral *E)
bool VisitSYCLUniqueStableNameExpr(const SYCLUniqueStableNameExpr *E)
bool visitCXXTryStmt(const CXXTryStmt *S)
static bool isUnevaluatedBuiltin(unsigned ID)
Unevaluated builtins don't get their arguments put on the stack automatically.
static bool shouldBeGloballyIndexed(const ValueDecl *VD)
Returns whether we should create a global variable for the given ValueDecl.
Scope used to handle temporaries in toplevel variable declarations.
DeclScope(Compiler< Emitter > *Ctx, const ValueDecl *VD)
Wrapper around fixed point types.
static FixedPoint zero(llvm::FixedPointSemantics Sem)
If a Floating is constructed from Memory, it DOES NOT OWN THAT MEMORY.
bool hasThisPointer() const
bool hasRVO() const
Checks if the first argument is a RVO pointer.
When generating code for e.g.
LocOverrideScope(Compiler< Emitter > *Ctx, SourceInfo NewValue, bool Enabled=true)
Generic scope for local variables.
bool destroyLocals(const Expr *E=nullptr) override
Explicit destruction of local variables.
LocalScope(Compiler< Emitter > *Ctx, ScopeKind Kind=ScopeKind::Block)
Sets the context for break/continue statements.
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
LoopScope(Compiler< Emitter > *Ctx, const Stmt *Name, LabelTy BreakLabel, LabelTy ContinueLabel)
PrimType value_or(PrimType PT) const
Scope used to handle initialization methods.
OptionScope(Compiler< Emitter > *Ctx, bool NewDiscardResult, bool NewInitializing, bool NewToLValue)
Root constructor, compiling or discarding primitives.
Context to manage declaration lifetimes.
Structure/Class descriptor.
bool isUnion() const
Checks if the record is a union.
const Field * getField(unsigned I) const
const Base * getBase(unsigned I) const
bool hasTrivialDtor() const
Returns true for anonymous unions and records with no destructor or for those with a trivial destruct...
Describes the statement/declaration an opcode was generated from.
StmtExprScope(Compiler< Emitter > *Ctx)
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
typename Compiler< Emitter >::CaseMap CaseMap
SwitchScope(Compiler< Emitter > *Ctx, const Stmt *Name, CaseMap &&CaseLabels, LabelTy BreakLabel, OptLabelTy DefaultLabel)
Scope chain managing the variable lifetimes.
Compiler< Emitter > * Ctx
Compiler instance.
virtual void addLocal(Scope::Local Local)
VariableScope * getParent() const
bool Sub(InterpState &S, CodePtr OpPC)
bool LT(InterpState &S, CodePtr OpPC)
static llvm::RoundingMode getRoundingMode(FPOptions FPO)
llvm::PointerUnion< const Decl *, const Expr * > DeclTy
constexpr bool isSignedType(PrimType T)
bool Div(InterpState &S, CodePtr OpPC)
1) Pops the RHS from the stack.
static bool Activate(InterpState &S, CodePtr OpPC)
constexpr bool isPtrType(PrimType T)
constexpr size_t align(size_t Size)
Aligns a size to the pointer alignment.
bool InitScope(InterpState &S, CodePtr OpPC, uint32_t I)
constexpr bool isIntegerOrBoolType(PrimType T)
static void discard(InterpStack &Stk, PrimType T)
bool LE(InterpState &S, CodePtr OpPC)
PrimType
Enumeration of the primitive types of the VM.
static std::optional< bool > getBoolValue(const Expr *E)
bool Init(InterpState &S, CodePtr OpPC)
bool Mul(InterpState &S, CodePtr OpPC)
size_t primSize(PrimType Type)
Returns the size of a primitive type in bytes.
bool Inc(InterpState &S, CodePtr OpPC, bool CanOverflow)
1) Pops a pointer from the stack 2) Load the value from the pointer 3) Writes the value increased by ...
bool Add(InterpState &S, CodePtr OpPC)
llvm::BitVector collectNonNullArgs(const FunctionDecl *F, ArrayRef< const Expr * > Args)
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
bool hasSpecificAttr(const Container &container)
@ Success
Annotation was successful.
UnaryExprOrTypeTrait
Names for the "expression or type" traits.
@ SD_Static
Static storage duration.
@ SD_FullExpression
Full-expression storage duration (for temporaries).
@ Result
The result type of a method or function.
OptionalUnsigned< unsigned > UnsignedOrNone
U cast(CodeGen::Address addr)
int const char * function
__packed_splat4 __packed_splat2 __packed_splat8 __packed_splat4 __packed_splat2 __packed_splat4 __packed_splat2 __packed_splat8 __packed_splat4 uint32_t
Describes a memory block created by an allocation site.
unsigned getNumElems() const
Returns the number of elements stored in the block.
bool isPrimitive() const
Checks if the descriptor is of a primitive.
QualType getElemQualType() const
bool hasTrivialDtor() const
Whether variables of this descriptor need their destructor called or not.
bool isCompositeArray() const
Checks if the descriptor is of an array of composites.
const Descriptor *const ElemDesc
Descriptor of the array element.
static constexpr MetadataSize InlineDescMD
bool isPrimitiveArray() const
Checks if the descriptor is of an array of primitives.
PrimType getPrimType() const
bool isRecord() const
Checks if the descriptor is of a record.
const Record *const ElemRecord
Pointer to the record, if block contains records.
bool isArray() const
Checks if the descriptor is of an array.
Descriptor used for global variables.
GlobalInitState InitState
static InitLink InitList()
static InitLink Elem(unsigned Index)
bool emit(Compiler< Emitter > *Ctx, const Expr *E) const
static InitLink Field(unsigned Offset)
static InitLink Decl(const ValueDecl *D)
static InitLink Temp(unsigned Offset)
Information about a local's storage.
State encapsulating if a the variable creation has been successful, unsuccessful, or no variable has ...
static VarCreationState NotCreated()