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;
179 : Ctx(Ctx), OldFlag(Ctx->LocOverride), Enabled(Enabled) {
182 Ctx->LocOverride = NewValue;
187 Ctx->LocOverride = OldFlag;
192 std::optional<SourceInfo> OldFlag;
199template <
class Emitter>
207 case CK_LValueToRValue: {
218 if (
const auto *DRE = dyn_cast<DeclRefExpr>(SubExpr)) {
224 if (
auto GlobalIndex =
P.getGlobal(D))
225 return this->emitGetGlobal(*SubExprT, *GlobalIndex, E);
226 }
else if (
auto It =
Locals.find(D); It !=
Locals.end()) {
227 return this->emitGetLocal(*SubExprT, It->second.Offset, E);
228 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
229 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
230 return this->emitGetParam(*SubExprT, It->second.Index, E);
242 if (!this->emitGetPtrLocal(*LocalIndex, E))
246 if (!this->
visit(SubExpr))
250 return this->emitLoadPop(*SubExprT, E);
255 return this->emitMemcpy(E);
258 case CK_DerivedToBaseMemberPointer: {
269 ->getMostRecentCXXRecordDecl();
271 const CXXRecordDecl *ToDecl = B->getType()->getAsCXXRecordDecl();
272 unsigned DerivedOffset =
Ctx.collectBaseOffset(ToDecl, CurDecl);
274 if (!this->emitCastMemberPtrBasePop(DerivedOffset, ToDecl, E))
282 case CK_BaseToDerivedMemberPointer: {
293 ->getMostRecentCXXRecordDecl();
297 typedef std::reverse_iterator<CastExpr::path_const_iterator> ReverseIter;
299 PathI != PathE; ++PathI) {
300 const CXXRecordDecl *ToDecl = (*PathI)->getType()->getAsCXXRecordDecl();
301 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
303 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
310 assert(ToDecl != CurDecl);
311 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
313 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
319 case CK_UncheckedDerivedToBase:
320 case CK_DerivedToBase: {
325 if (
const auto *PT = dyn_cast<PointerType>(Ty))
326 return PT->getPointeeType()->getAsCXXRecordDecl();
327 return Ty->getAsCXXRecordDecl();
334 if (B->isVirtual()) {
335 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), E))
337 CurType = B->getType();
339 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
340 if (!this->emitGetPtrBasePop(
343 CurType = B->getType();
350 case CK_BaseToDerived: {
353 unsigned DerivedOffset =
359 return this->emitGetPtrDerivedPop(DerivedOffset,
364 case CK_FloatingCast: {
367 return this->emitVectorConversion(E->
getSubExpr(), E);
371 if (!this->
visit(SubExpr))
373 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
377 case CK_IntegralToFloating: {
379 return this->emitVectorConversion(E->
getSubExpr(), E);
382 if (!this->
visit(SubExpr))
384 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
385 return this->emitCastIntegralFloating(
classifyPrim(SubExpr),
386 TargetSemantics, getFPOptions(E), E);
389 case CK_FloatingToBoolean: {
391 return this->emitVectorConversion(E->
getSubExpr(), E);
395 if (
const auto *FL = dyn_cast<FloatingLiteral>(SubExpr))
396 return this->emitConstBool(FL->getValue().isNonZero(), E);
397 if (!this->
visit(SubExpr))
399 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
402 case CK_FloatingToIntegral: {
404 return this->emitVectorConversion(E->
getSubExpr(), E);
407 if (!this->
visit(SubExpr))
411 return this->emitCastFloatingIntegralAP(
Ctx.getBitWidth(E->
getType()),
414 return this->emitCastFloatingIntegralAPS(
Ctx.getBitWidth(E->
getType()),
417 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
420 case CK_NullToPointer:
421 case CK_NullToMemberPointer: {
424 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
429 case CK_PointerToIntegral: {
430 if (!this->
visit(SubExpr))
436 if (!this->emitDecayPtr(FromT,
PT_Ptr, E))
442 return this->emitCastPointerIntegralAP(
Ctx.getBitWidth(E->
getType()), E);
444 return this->emitCastPointerIntegralAPS(
Ctx.getBitWidth(E->
getType()), E);
445 return this->emitCastPointerIntegral(T, E);
448 case CK_ArrayToPointerDecay: {
449 if (!this->
visit(SubExpr))
451 return this->emitArrayDecay(E);
454 case CK_IntegralToPointer: {
456 assert(IntType->isIntegralOrEnumerationType());
457 if (!this->
visit(SubExpr))
471 return this->emitDecayPtr(
PT_Ptr, DestPtrT, E);
474 case CK_AtomicToNonAtomic:
475 case CK_ConstructorConversion:
476 case CK_FunctionToPointerDecay:
477 case CK_NonAtomicToAtomic:
479 case CK_UserDefinedConversion:
480 case CK_AddressSpaceConversion:
481 case CK_CPointerToObjCPointerCast:
498 return this->emitBuiltinBitCast(E);
513 if (!this->
visit(SubExpr))
521 return this->emitFnPtrCast(E);
528 if (!this->
visit(SubExpr))
530 return this->emitDecayPtr(*FromT, *ToT, E);
532 case CK_IntegralToBoolean:
533 case CK_FixedPointToBoolean: {
535 return this->emitVectorConversion(E->
getSubExpr(), E);
541 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr))
542 return this->emitConst(IL->getValue(), E);
543 if (!this->
visit(SubExpr))
548 case CK_IntegralCast:
550 return this->emitVectorConversion(E->
getSubExpr(), E);
552 case CK_BooleanToSignedIntegral: {
559 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr)) {
564 if (!this->emitConst(IL->getValue(), SubExpr))
567 if (!this->
visit(SubExpr))
575 if (!ED->isFixed()) {
576 if (!this->emitCheckEnumValue(*FromT, ED, E))
582 if (!this->emitCastAP(*FromT,
Ctx.getBitWidth(E->
getType()), E))
585 if (!this->emitCastAPS(*FromT,
Ctx.getBitWidth(E->
getType()), E))
590 if (!this->emitCast(*FromT, *ToT, E))
593 if (E->
getCastKind() == CK_BooleanToSignedIntegral)
594 return this->emitNeg(*ToT, E);
598 case CK_PointerToBoolean:
599 case CK_MemberPointerToBoolean: {
602 if (!this->
visit(SubExpr))
604 return this->emitIsNonNull(PtrT, E);
607 case CK_IntegralComplexToBoolean:
608 case CK_FloatingComplexToBoolean: {
609 if (!this->
visit(SubExpr))
611 return this->emitComplexBoolCast(SubExpr);
614 case CK_IntegralComplexToReal:
615 case CK_FloatingComplexToReal:
616 return this->emitComplexReal(SubExpr);
618 case CK_IntegralRealToComplex:
619 case CK_FloatingRealToComplex: {
626 if (!this->emitGetPtrLocal(*LocalIndex, E))
635 if (!this->visitZeroInitializer(T, SubExpr->
getType(), SubExpr))
637 return this->emitInitElem(T, 1, SubExpr);
640 case CK_IntegralComplexCast:
641 case CK_FloatingComplexCast:
642 case CK_IntegralComplexToFloatingComplex:
643 case CK_FloatingComplexToIntegralComplex: {
650 if (!this->emitGetPtrLocal(*LocalIndex, E))
657 unsigned SubExprOffset =
659 if (!this->
visit(SubExpr))
661 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
669 for (
unsigned I = 0; I != 2; ++I) {
670 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, E))
672 if (!this->emitArrayElemPop(SourceElemT, I, E))
676 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, E))
680 if (!this->emitInitElem(DestElemT, I, E))
686 case CK_VectorSplat: {
697 if (!this->emitGetPtrLocal(*LocalIndex, E))
703 unsigned ElemOffset =
707 if (!this->
visit(SubExpr))
712 if (!this->emitSetLocal(ElemT, ElemOffset, E))
715 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
716 if (!this->emitGetLocal(ElemT, ElemOffset, E))
718 if (!this->emitInitElem(ElemT, I, E))
725 case CK_HLSLVectorTruncation: {
730 if (!this->
visit(SubExpr))
732 return this->emitArrayElemPop(*ResultT, 0, E);
741 if (!this->emitGetPtrLocal(*LocalIndex, E))
746 if (!this->
visit(SubExpr))
748 return this->emitCopyArray(classifyVectorElementType(E->
getType()), 0, 0,
752 case CK_IntegralToFixedPoint: {
753 if (!this->
visit(SubExpr))
757 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
762 return this->emitPopFixedPoint(E);
765 case CK_FloatingToFixedPoint: {
766 if (!this->
visit(SubExpr))
770 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
771 if (!this->emitCastFloatingFixedPoint(Sem, E))
774 return this->emitPopFixedPoint(E);
777 case CK_FixedPointToFloating: {
778 if (!this->
visit(SubExpr))
780 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
781 if (!this->emitCastFixedPointFloating(TargetSemantics, E))
784 return this->emitPopFloat(E);
787 case CK_FixedPointToIntegral: {
788 if (!this->
visit(SubExpr))
791 if (!this->emitCastFixedPointIntegral(IntegralT, E))
794 return this->emitPop(IntegralT, E);
797 case CK_FixedPointCast: {
798 if (!this->
visit(SubExpr))
801 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
802 if (!this->emitCastFixedPoint(Sem, E))
805 return this->emitPopFixedPoint(E);
813 llvm_unreachable(
"CXXDynamicCastExpr has its own function");
815 case CK_LValueBitCast:
820 case CK_HLSLArrayRValue: {
827 if (!this->emitGetPtrLocal(*LocalIndex, E))
830 if (!this->
visit(SubExpr))
832 return this->emitMemcpy(E);
835 case CK_HLSLMatrixTruncation: {
840 if (!this->
visit(SubExpr))
842 return this->emitArrayElemPop(*ResultT, 0, E);
851 if (!this->emitGetPtrLocal(*LocalIndex, E))
856 if (!this->
visit(SubExpr))
858 return this->emitCopyArray(classifyMatrixElementType(SubExpr->
getType()), 0,
862 case CK_HLSLAggregateSplatCast: {
872 if (!this->emitGetPtrLocal(*LocalIndex, E))
882 if (!this->
visit(SubExpr))
884 if (!this->emitSetLocal(SrcElemT, SrcOffset, E))
888 return emitHLSLAggregateSplat(SrcElemT, SrcOffset, E->
getType(), E);
891 case CK_HLSLElementwiseCast: {
902 unsigned SrcPtrOffset =
904 if (!this->
visit(SubExpr))
906 if (!this->emitSetLocal(
PT_Ptr, SrcPtrOffset, E))
910 if (!emitHLSLFlattenAggregate(SrcType, SrcPtrOffset, Elements, 1, E))
912 if (Elements.empty())
915 const HLSLFlatElement &Src = Elements[0];
916 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
918 return this->emitPrimCast(Src.Type, *DestT, DestType, E);
925 if (!this->emitGetPtrLocal(*LocalIndex, E))
931 if (!this->
visit(SubExpr))
933 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
937 unsigned ElemCount = countHLSLFlatElements(DestType);
940 Elements.reserve(ElemCount);
941 if (!emitHLSLFlattenAggregate(SrcType, SrcOffset, Elements, ElemCount, E))
946 assert(Elements.size() == ElemCount &&
947 "Source type has fewer scalar elements than the destination type");
949 return emitHLSLConstructAggregate(DestType, Elements, E);
956 const Record::Field *RF = R->getField(UnionField);
957 QualType FieldType = RF->Decl->getType();
960 if (!this->
visit(SubExpr))
962 if (RF->isBitField())
963 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
965 return this->emitInitFieldActivate(*PT, RF->Offset, E);
968 if (!this->emitGetPtrField(RF->Offset, E))
970 if (!this->emitActivate(E))
976 return this->emitInvalid(E);
978 llvm_unreachable(
"Unhandled clang::CastKind enum");
981template <
class Emitter>
983 return this->emitBuiltinBitCast(E);
986template <
class Emitter>
991 return this->emitConst(
LE->getValue(),
LE);
994template <
class Emitter>
1000 return this->emitFloat(F, E);
1003template <
class Emitter>
1013 if (!this->emitGetPtrLocal(*LocalIndex, E))
1020 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
1022 if (!this->emitInitElem(SubExprT, 0, SubExpr))
1027template <
class Emitter>
1035 auto Sem =
Ctx.getASTContext().getFixedPointSemantics(E->
getType());
1040template <
class Emitter>
1045template <
class Emitter>
1072 return this->emitComplexComparison(LHS, RHS, E);
1080 if (!this->
visit(LHS))
1083 if (!this->
visit(RHS))
1086 if (!this->emitToMemberPtr(E))
1092 if (!this->emitCastMemberPtrPtr(E))
1109 Ctx.getASTContext().CompCategories.lookupInfoForType(E->
getType());
1115 if (!this->emitGetPtrLocal(*ResultIndex, E))
1122 return this->emitCMP3(*
LT, CmpInfo, E);
1125 if (!
LT || !RT || !T)
1135 return this->visitAssignment(LHS, RHS, E);
1142 auto MaybeCastToBool = [
this, T, E](
bool Result) {
1146 return this->emitPopBool(E);
1148 return this->emitCast(
PT_Bool, *T, E);
1152 auto Discard = [
this, T, E](
bool Result) {
1160 return MaybeCastToBool(this->emitEQ(*
LT, E));
1162 return MaybeCastToBool(this->emitNE(*
LT, E));
1164 return MaybeCastToBool(this->emitLT(*
LT, E));
1166 return MaybeCastToBool(this->emitLE(*
LT, E));
1168 return MaybeCastToBool(this->emitGT(*
LT, E));
1170 return MaybeCastToBool(this->emitGE(*
LT, E));
1173 return Discard(this->emitSubf(getFPOptions(E), E));
1174 return Discard(this->emitSub(*T, E));
1177 return Discard(this->emitAddf(getFPOptions(E), E));
1178 return Discard(this->emitAdd(*T, E));
1181 return Discard(this->emitMulf(getFPOptions(E), E));
1182 return Discard(this->emitMul(*T, E));
1184 return Discard(this->emitRem(*T, E));
1187 return Discard(this->emitDivf(getFPOptions(E), E));
1188 return Discard(this->emitDiv(*T, E));
1190 return Discard(this->emitBitAnd(*T, E));
1192 return Discard(this->emitBitOr(*T, E));
1194 return Discard(this->emitShl(*
LT, *RT, E));
1196 return Discard(this->emitShr(*
LT, *RT, E));
1198 return Discard(this->emitBitXor(*T, E));
1201 llvm_unreachable(
"Already handled earlier");
1206 llvm_unreachable(
"Unhandled binary op");
1211template <
class Emitter>
1217 if ((Op != BO_Add && Op != BO_Sub) ||
1228 auto visitAsPointer = [&](
const Expr *E,
PrimType T) ->
bool {
1229 if (!this->
visit(E))
1232 return this->emitDecayPtr(T,
PT_Ptr, E);
1241 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1249 ElemTypeSize =
Ctx.getASTContext().getTypeSizeInChars(ElemType);
1252 if (!this->emitSubPtr(IntT, ElemTypeSize.
isZero(), E))
1259 if (!visitAsPointer(RHS, *RT))
1261 if (!this->
visit(LHS))
1265 if (!visitAsPointer(LHS, *
LT))
1267 if (!this->
visit(RHS))
1278 if (!this->emitAddOffset(OffsetType, E))
1282 if (!this->emitSubOffset(OffsetType, E))
1299template <
class Emitter>
1309 LabelTy LabelTrue = this->getLabel();
1310 LabelTy LabelEnd = this->getLabel();
1314 if (!this->jumpTrue(LabelTrue, E))
1319 if (!this->jump(LabelEnd, E))
1322 this->emitLabel(LabelTrue);
1323 this->emitConstBool(
true, E);
1324 this->fallthrough(LabelEnd);
1325 this->emitLabel(LabelEnd);
1328 assert(Op == BO_LAnd);
1331 LabelTy LabelFalse = this->getLabel();
1332 LabelTy LabelEnd = this->getLabel();
1336 if (!this->jumpFalse(LabelFalse, E))
1341 if (!this->jump(LabelEnd, E))
1344 this->emitLabel(LabelFalse);
1345 this->emitConstBool(
false, E);
1346 this->fallthrough(LabelEnd);
1347 this->emitLabel(LabelEnd);
1351 return this->emitPopBool(E);
1356 return this->emitCast(
PT_Bool, *T, E);
1360template <
class Emitter>
1367 if (!this->emitGetPtrLocal(*LocalIndex, E))
1376 PrimType ResultElemT = this->classifyComplexElementType(E->
getType());
1377 unsigned ResultOffset = ~0u;
1383 if (!this->emitDupPtr(E))
1385 if (!this->emitSetLocal(
PT_Ptr, ResultOffset, E))
1390 LHSType = AT->getValueType();
1393 RHSType = AT->getValueType();
1402 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1407 if (!this->
visit(LHS))
1409 if (!this->
visit(RHS))
1411 if (!this->emitMulc(ElemT, E))
1414 return this->emitPopPtr(E);
1418 if (Op == BO_Div && RHSIsComplex) {
1425 if (!LHSIsComplex) {
1430 LHSOffset = *LocalIndex;
1432 if (!this->emitGetPtrLocal(LHSOffset, E))
1435 if (!this->
visit(LHS))
1438 if (!this->emitInitElem(ElemT, 0, E))
1441 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1443 if (!this->emitInitElem(ElemT, 1, E))
1446 if (!this->
visit(LHS))
1450 if (!this->
visit(RHS))
1452 if (!this->emitDivc(ElemT, E))
1455 return this->emitPopPtr(E);
1462 if (!this->
visit(LHS))
1464 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1469 if (!this->
visit(LHS))
1471 if (!this->emitSetLocal(LHST, LHSOffset, E))
1479 if (!this->
visit(RHS))
1481 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1486 if (!this->
visit(RHS))
1488 if (!this->emitSetLocal(RHST, RHSOffset, E))
1495 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1496 unsigned ElemIndex,
unsigned Offset,
1497 const Expr *E) ->
bool {
1499 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1501 return this->emitArrayElemPop(classifyComplexElementType(E->
getType()),
1504 if (ElemIndex == 0 || !LoadZero)
1511 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1514 if (!this->emitGetLocal(
PT_Ptr, ResultOffset, E))
1521 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1524 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1527 if (!this->emitAddf(getFPOptions(E), E))
1530 if (!this->emitAdd(ResultElemT, E))
1535 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1538 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1541 if (!this->emitSubf(getFPOptions(E), E))
1544 if (!this->emitSub(ResultElemT, E))
1549 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1552 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1556 if (!this->emitMulf(getFPOptions(E), E))
1559 if (!this->emitMul(ResultElemT, E))
1564 assert(!RHSIsComplex);
1565 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1568 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1572 if (!this->emitDivf(getFPOptions(E), E))
1575 if (!this->emitDiv(ResultElemT, E))
1586 if (!this->emitInitElemPop(ResultElemT, ElemIndex, E))
1589 if (!this->emitPop(ResultElemT, E))
1594 return this->emitPopPtr(E);
1600template <
class Emitter>
1605 "Comma op should be handled in VisitBinaryOperator");
1619 if (!this->emitGetPtrLocal(*LocalIndex, E))
1633 assert(
Ctx.getASTContext().hasSameUnqualifiedType(
1636 if (!this->
visit(LHS))
1638 if (!this->
visit(RHS))
1640 if (!this->emitCopyArray(ElemT, 0, 0, VecTy->getNumElements(), E))
1643 return this->emitPopPtr(E);
1648 unsigned LHSOffset =
1650 if (!this->
visit(LHS))
1652 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1656 unsigned RHSOffset =
1658 if (!this->
visit(RHS))
1660 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1672 if (NeedIntPromot) {
1674 Ctx.getASTContext().getPromotedIntegerType(
Ctx.getASTContext().BoolTy);
1679 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1680 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1682 if (!this->emitArrayElemPop(ElemT, Index, E))
1685 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
1687 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1689 }
else if (NeedIntPromot) {
1690 if (!this->emitPrimCast(ElemT, PromotT, PromotTy, E))
1696#define EMIT_ARITH_OP(OP) \
1698 if (ElemT == PT_Float) { \
1699 if (!this->emit##OP##f(getFPOptions(E), E)) \
1702 if (!this->emit##OP(ElemT, E)) \
1708 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1709 if (!getElem(LHSOffset, ElemT, I))
1711 if (!getElem(RHSOffset, RHSElemT, I))
1723 if (!this->emitRem(ElemT, E))
1727 if (!this->emitBitAnd(OpT, E))
1731 if (!this->emitBitOr(OpT, E))
1735 if (!this->emitBitXor(OpT, E))
1739 if (!this->emitShl(OpT, RHSElemT, E))
1743 if (!this->emitShr(OpT, RHSElemT, E))
1747 if (!this->emitEQ(ElemT, E))
1751 if (!this->emitNE(ElemT, E))
1755 if (!this->emitLE(ElemT, E))
1759 if (!this->emitLT(ElemT, E))
1763 if (!this->emitGE(ElemT, E))
1767 if (!this->emitGT(ElemT, E))
1772 if (!this->emitBitAnd(ResultElemT, E))
1777 if (!this->emitBitOr(ResultElemT, E))
1781 return this->emitInvalid(E);
1790 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1792 if (!this->emitNeg(ResultElemT, E))
1798 if (NeedIntPromot &&
1799 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(), E))
1803 if (!this->emitInitElem(ResultElemT, I, E))
1812template <
class Emitter>
1822 auto LHSSemaInt = LHSSema.toOpaqueInt();
1824 auto RHSSemaInt = RHSSema.toOpaqueInt();
1826 if (!this->
visit(LHS))
1834 if (!this->
visit(RHS))
1843 auto ConvertResult = [&](
bool R) ->
bool {
1847 auto CommonSema = LHSSema.getCommonSemantics(RHSSema).toOpaqueInt();
1848 if (ResultSema != CommonSema)
1849 return this->emitCastFixedPoint(ResultSema, E);
1853 auto MaybeCastToBool = [&](
bool Result) {
1858 return this->emitPop(T, E);
1860 return this->emitCast(
PT_Bool, T, E);
1866 return MaybeCastToBool(this->emitEQFixedPoint(E));
1868 return MaybeCastToBool(this->emitNEFixedPoint(E));
1870 return MaybeCastToBool(this->emitLTFixedPoint(E));
1872 return MaybeCastToBool(this->emitLEFixedPoint(E));
1874 return MaybeCastToBool(this->emitGTFixedPoint(E));
1876 return MaybeCastToBool(this->emitGEFixedPoint(E));
1878 return ConvertResult(this->emitAddFixedPoint(E));
1880 return ConvertResult(this->emitSubFixedPoint(E));
1882 return ConvertResult(this->emitMulFixedPoint(E));
1884 return ConvertResult(this->emitDivFixedPoint(E));
1886 return ConvertResult(this->emitShiftFixedPoint(
true, E));
1888 return ConvertResult(this->emitShiftFixedPoint(
false, E));
1891 return this->emitInvalid(E);
1894 llvm_unreachable(
"unhandled binop opcode");
1897template <
class Emitter>
1906 if (!this->
visit(SubExpr))
1908 if (!this->emitNegFixedPoint(E))
1911 return this->emitPopFixedPoint(E);
1917 llvm_unreachable(
"Unhandled unary opcode");
1920template <
class Emitter>
1929 return this->visitZeroInitializer(*T, QT, E);
1937 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1938 CXXRD && CXXRD->getNumVBases() > 0) {
1948 return this->visitZeroRecordInitializer(R, E);
1955 return this->visitZeroArrayInitializer(QT, E);
1959 QualType ElemQT = ComplexTy->getElementType();
1961 for (
unsigned I = 0; I < 2; ++I) {
1962 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1964 if (!this->emitInitElem(ElemT, I, E))
1971 unsigned NumVecElements = VecT->getNumElements();
1972 QualType ElemQT = VecT->getElementType();
1975 for (
unsigned I = 0; I < NumVecElements; ++I) {
1976 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1978 if (!this->emitInitElem(ElemT, I, E))
1985 unsigned NumElems = MT->getNumElementsFlattened();
1986 QualType ElemQT = MT->getElementType();
1989 for (
unsigned I = 0; I != NumElems; ++I) {
1990 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1992 if (!this->emitInitElem(ElemT, I, E))
2001template <
class Emitter>
2014 for (
const Expr *SubExpr : {LHS, RHS}) {
2015 if (!this->
visit(SubExpr)) {
2022 if (SubExpr ==
Base &&
Base->getType()->isPointerType()) {
2023 if (!this->emitExpandPtr(E))
2034 return this->emitError(E);
2037 if (!this->emitFlip(
PT_Ptr, *IndexT, E))
2041 if (!this->emitArrayElemPtrPop(*IndexT, E))
2044 return this->emitPopPtr(E);
2050 return this->emitLoadPop(*T, E);
2053template <
class Emitter>
2055 const Expr *ArrayFiller,
const Expr *E) {
2060 QT = AT->getValueType();
2063 if (
Inits.size() == 0)
2065 return this->emitInvalid(E);
2080 if (
Inits.size() == 0)
2081 return this->visitZeroInitializer(*T, QT, E);
2082 assert(
Inits.size() == 1);
2095 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
2102 bool BitField = FieldToInit->isBitField();
2104 return this->emitInitBitFieldActivate(T, FieldToInit->Offset,
2105 FieldToInit->bitWidth(), E);
2107 return this->emitInitBitField(T, FieldToInit->Offset,
2108 FieldToInit->bitWidth(), E);
2110 return this->emitInitFieldActivate(T, FieldToInit->Offset, E);
2111 return this->emitInitField(T, FieldToInit->Offset, E);
2114 auto initCompositeField = [=](
const Record::Field *FieldToInit,
2122 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
2125 if (
Activate && !this->emitActivate(E))
2132 if (
Inits.size() == 0) {
2133 if (!this->visitZeroRecordInitializer(R, E))
2138 if (
const auto *ILE = dyn_cast<InitListExpr>(E))
2139 FToInit = ILE->getInitializedFieldInUnion();
2143 const Record::Field *FieldToInit = R->getField(FToInit);
2145 if (!initPrimitiveField(FieldToInit,
Init, *T,
true))
2148 if (!initCompositeField(FieldToInit,
Init,
true))
2152 return this->emitFinishInit(E);
2155 assert(!R->isUnion());
2156 unsigned InitIndex = 0;
2159 while (InitIndex < R->getNumFields() &&
2160 R->getField(InitIndex)->isUnnamedBitField())
2171 const Record::Field *FieldToInit = R->getField(InitIndex);
2172 if (!initPrimitiveField(FieldToInit,
Init, *T))
2177 if (
const Record::Base *B = R->getBase(
Init->getType())) {
2178 if (!this->emitGetPtrBase(B->Offset,
Init))
2186 const Record::Field *FieldToInit = R->getField(InitIndex);
2187 if (!initCompositeField(FieldToInit,
Init))
2193 return this->emitFinishInit(E);
2198 Ctx.getASTContext().getAsConstantArrayType(QT);
2201 if (
Initializing && !this->emitCheckArrayDestSize(NumElems, E))
2204 if (
Inits.size() == 1 && QT ==
Inits[0]->getType())
2208 unsigned ElementIndex = 0;
2210 if (
const auto *EmbedS =
2211 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
2219 if (!this->emitCastIntegralFloating(
classifyPrim(IL), Sem,
2220 getFPOptions(E), E))
2226 return this->emitInitElem(TargetT, ElemIndex, IL);
2228 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
2244 for (; ElementIndex != NumElems; ++ElementIndex) {
2250 return this->emitFinishInit(E);
2254 unsigned NumInits =
Inits.size();
2259 QualType ElemQT = ComplexTy->getElementType();
2261 if (NumInits == 0) {
2263 for (
unsigned I = 0; I < 2; ++I) {
2264 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2266 if (!this->emitInitElem(ElemT, I, E))
2269 }
else if (NumInits == 2) {
2270 unsigned InitIndex = 0;
2275 if (!this->emitInitElem(ElemT, InitIndex, E))
2284 unsigned NumVecElements = VecT->getNumElements();
2285 assert(NumVecElements >=
Inits.size());
2287 QualType ElemQT = VecT->getElementType();
2291 unsigned InitIndex = 0;
2298 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
2299 if (!this->emitCopyArray(ElemT, 0, InitIndex,
2300 InitVecT->getNumElements(), E))
2302 InitIndex += InitVecT->getNumElements();
2304 if (!this->emitInitElem(ElemT, InitIndex, E))
2310 assert(InitIndex <= NumVecElements);
2313 for (; InitIndex != NumVecElements; ++InitIndex) {
2314 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2316 if (!this->emitInitElem(ElemT, InitIndex, E))
2323 unsigned NumElems = MT->getNumElementsFlattened();
2324 assert(
Inits.size() == NumElems);
2326 QualType ElemQT = MT->getElementType();
2332 for (
unsigned I = 0; I != NumElems; ++I) {
2335 if (!this->emitInitElem(ElemT, I, E))
2346template <
class Emitter>
2353 return this->emitInitElem(*InitT, ElemIndex,
Init);
2359 if (!this->emitConstUint32(ElemIndex,
Init))
2361 if (!this->emitArrayElemPtrUint32(
Init))
2366template <
class Emitter>
2369 bool Activate,
bool IsOperatorCall) {
2371 llvm::BitVector NonNullArgs;
2372 if (FuncDecl && FuncDecl->
hasAttr<NonNullAttr>())
2375 bool ExplicitMemberFn =
false;
2376 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl))
2377 ExplicitMemberFn = MD->isExplicitObjectMemberFunction();
2379 unsigned ArgIndex = 0;
2380 for (
const Expr *Arg : Args) {
2382 if (!this->
visit(Arg))
2390 unsigned DeclIndex = ArgIndex - IsOperatorCall + ExplicitMemberFn;
2391 if (DeclIndex < FuncDecl->getNumParams())
2392 Source = FuncDecl->
getParamDecl(ArgIndex - IsOperatorCall +
2401 if (!this->emitGetPtrLocal(*LocalIndex, Arg))
2409 if (!this->emitActivate(Arg))
2413 if (!NonNullArgs.empty() && NonNullArgs[ArgIndex]) {
2416 if (!this->emitCheckNonNullArg(ArgT, Arg))
2427template <
class Emitter>
2432template <
class Emitter>
2438template <
class Emitter>
2444template <
class Emitter>
2462template <
class Emitter>
2464 auto It = E->
begin();
2465 return this->
visit(*It);
2470 bool AlignOfReturnsPreferred =
2477 T = Ref->getPointeeType();
2479 if (T.getQualifiers().hasUnaligned())
2485 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2491template <
class Emitter>
2498 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2504 ArgType = Ref->getPointeeType();
2510 if (
ArgType->isDependentType() || !
ArgType->isConstantSizeType())
2511 return this->emitInvalid(E);
2513 if (Kind == UETT_SizeOf)
2522 return this->emitConst(Size.getQuantity(), E);
2525 if (Kind == UETT_CountOf) {
2531 if (
const auto *CAT =
2535 return this->emitConst(CAT->getSize(), E);
2545 if (VAT->getElementType()->isArrayType()) {
2546 std::optional<APSInt> Res =
2548 ? VAT->getSizeExpr()->getIntegerConstantExpr(ASTCtx)
2553 return this->emitConst(*Res, E);
2558 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2578 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2581 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2591 return this->emitConst(Size.getQuantity(), E);
2594 if (Kind == UETT_VectorElements) {
2599 return this->emitConst(VT->getNumElements(), E);
2601 return this->emitSizelessVectorElementSize(E);
2604 if (Kind == UETT_VecStep) {
2606 unsigned N = VT->getNumElements();
2613 return this->emitConst(N, E);
2615 return this->emitConst(1, E);
2618 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2627 if (Kind == UETT_PtrAuthTypeDiscriminator) {
2629 return this->emitInvalid(E);
2631 return this->emitConst(
2632 const_cast<ASTContext &
>(ASTCtx).getPointerAuthTypeDiscriminator(
2640template <
class Emitter>
2649 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2652 if (
auto GlobalIndex =
P.getGlobal(VD)) {
2653 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
2655 if (
Member->getType()->isReferenceType())
2656 return this->emitLoadPopPtr(E);
2665 if (
const auto *MD = dyn_cast<CXXMethodDecl>(
Member);
2666 MD && !MD->isStatic()) {
2670 if (!this->
discard(Base) && !this->emitSideEffect(E))
2685 const Record::Field *F = R->getField(FD);
2689 const auto maybeLoadValue = [&]() ->
bool {
2693 return this->emitLoadPop(*T, E);
2698 if (F->Decl->getType()->isReferenceType())
2699 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2700 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2703template <
class Emitter>
2713template <
class Emitter>
2727 if (!this->
visit(Common))
2729 return this->emitCopyArray(*SubExprT, 0, 0, Size, E);
2743 for (
size_t I = 0; I != Size; ++I) {
2755template <
class Emitter>
2770 return this->emitGetLocal(SubExprT, It->second, E);
2773 if (!this->
visit(SourceExpr))
2780 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2789 return this->emitGetLocal(SubExprT, LocalIndex, E);
2793template <
class Emitter>
2806 bool IsBcpCall =
false;
2807 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2808 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2812 LabelTy LabelEnd = this->getLabel();
2813 LabelTy LabelFalse = this->getLabel();
2816 if (!this->emitPushIgnoreDiags(E))
2824 if (this->checkingForUndefinedBehavior()) {
2827 if (!this->
discard(FalseExpr))
2844 if (!this->jumpFalse(LabelFalse, E))
2849 if (!this->jump(LabelEnd, E))
2851 this->emitLabel(LabelFalse);
2855 this->fallthrough(LabelEnd);
2856 this->emitLabel(LabelEnd);
2859 return this->emitPopIgnoreDiags(E);
2863template <
class Emitter>
2869 unsigned StringIndex =
P.createGlobalString(E);
2870 return this->emitGetPtrGlobal(StringIndex, E);
2875 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2876 assert(CAT &&
"a string literal that's not a constant array?");
2881 unsigned N = std::min(ArraySize, E->
getLength());
2884 for (
unsigned I = 0; I != N; ++I) {
2887 if (CharWidth == 1) {
2888 this->emitConstSint8(CodeUnit, E);
2889 this->emitInitElemSint8(I, E);
2890 }
else if (CharWidth == 2) {
2891 this->emitConstUint16(CodeUnit, E);
2892 this->emitInitElemUint16(I, E);
2893 }
else if (CharWidth == 4) {
2894 this->emitConstUint32(CodeUnit, E);
2895 this->emitInitElemUint32(I, E);
2897 llvm_unreachable(
"unsupported character width");
2902 for (
unsigned I = N; I != ArraySize; ++I) {
2903 if (CharWidth == 1) {
2904 this->emitConstSint8(0, E);
2905 this->emitInitElemSint8(I, E);
2906 }
else if (CharWidth == 2) {
2907 this->emitConstUint16(0, E);
2908 this->emitInitElemUint16(I, E);
2909 }
else if (CharWidth == 4) {
2910 this->emitConstUint32(0, E);
2911 this->emitInitElemUint32(I, E);
2913 llvm_unreachable(
"unsupported character width");
2920template <
class Emitter>
2924 return this->emitDummyPtr(E, E);
2927template <
class Emitter>
2929 auto &A =
Ctx.getASTContext();
2938template <
class Emitter>
2946 auto &A =
Ctx.getASTContext();
2950 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2951 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2958 unsigned StringIndex =
P.createGlobalString(SL);
2959 return this->emitGetPtrGlobal(StringIndex, E);
2962template <
class Emitter>
2966 return this->emitConst(E->
getValue(), E);
2969template <
class Emitter>
2995 if (!this->emitSetLocal(*RT, TempOffset, E))
3001 if (!this->emitLoad(LHST, E))
3005 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
3006 LHSComputationType, E))
3010 if (!this->emitGetLocal(*RT, TempOffset, E))
3015 if (!this->emitAddf(getFPOptions(E), E))
3019 if (!this->emitSubf(getFPOptions(E), E))
3023 if (!this->emitMulf(getFPOptions(E), E))
3027 if (!this->emitDivf(getFPOptions(E), E))
3038 return this->emitStorePop(LHST, E);
3039 return this->emitStore(LHST, E);
3042template <
class Emitter>
3051 if (Op != BO_AddAssign && Op != BO_SubAssign)
3060 if (!this->emitLoad(*
LT, LHS))
3066 if (Op == BO_AddAssign) {
3067 if (!this->emitAddOffset(*RT, E))
3070 if (!this->emitSubOffset(*RT, E))
3075 return this->emitStorePopPtr(E);
3076 return this->emitStorePtr(E);
3079template <
class Emitter>
3092 if (!
Ctx.getLangOpts().CPlusPlus14)
3093 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
3095 if (!
LT || !RT || !ResultT || !LHSComputationT)
3120 if (!this->emitSetLocal(*RT, TempOffset, E))
3127 if (!this->emitLoad(*
LT, E))
3129 if (
LT != LHSComputationT &&
3135 if (!this->emitGetLocal(*RT, TempOffset, E))
3141 if (!this->emitAdd(*LHSComputationT, E))
3145 if (!this->emitSub(*LHSComputationT, E))
3149 if (!this->emitMul(*LHSComputationT, E))
3153 if (!this->emitDiv(*LHSComputationT, E))
3157 if (!this->emitRem(*LHSComputationT, E))
3161 if (!this->emitShl(*LHSComputationT, *RT, E))
3165 if (!this->emitShr(*LHSComputationT, *RT, E))
3169 if (!this->emitBitAnd(*LHSComputationT, E))
3173 if (!this->emitBitXor(*LHSComputationT, E))
3177 if (!this->emitBitOr(*LHSComputationT, E))
3181 llvm_unreachable(
"Unimplemented compound assign operator");
3185 if (ResultT != LHSComputationT &&
3186 !this->emitIntegralCast(*LHSComputationT, *ResultT, E->
getType(), E))
3192 return this->emitStoreBitFieldPop(*ResultT, E);
3193 return this->emitStorePop(*ResultT, E);
3196 return this->emitStoreBitField(*ResultT, E);
3197 return this->emitStore(*ResultT, E);
3200template <
class Emitter>
3208template <
class Emitter>
3223 if (!
Ctx.getLangOpts().CPlusPlus11)
3230 for (
const Expr *LHS : CommaLHSs) {
3252 if (!this->
visit(Inner))
3257 if (!this->emitInitGlobalTemp(*InnerT, *GlobalIndex, TempDecl, E))
3260 if (!this->emitInitGlobal(*InnerT, *GlobalIndex, E))
3263 return this->emitGetPtrGlobal(*GlobalIndex, E);
3266 if (!this->checkLiteralType(Inner))
3269 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3275 return this->emitInitGlobalTempComp(TempDecl, E);
3288 unsigned LocalIndex =
3290 if (!this->VarScope->LocalsAlwaysEnabled &&
3291 !this->emitEnableLocal(LocalIndex, E))
3294 if (!this->
visit(Inner))
3296 if (!this->emitSetLocal(*InnerT, LocalIndex, E))
3299 return this->emitGetPtrLocal(LocalIndex, E);
3302 if (!this->checkLiteralType(Inner))
3309 if (!this->VarScope->LocalsAlwaysEnabled &&
3310 !this->emitEnableLocal(*LocalIndex, E))
3313 if (!this->emitGetPtrLocal(*LocalIndex, E))
3320template <
class Emitter>
3331 if (!this->
visit(SubExpr))
3335 return this->emitPopPtr(E);
3339template <
class Emitter>
3360 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3365 if (
P.isGlobalInitialized(*GlobalIndex))
3371 return this->emitInitGlobal(*T, *GlobalIndex, E);
3383 unsigned LocalIndex;
3387 LocalIndex = *MaybeIndex;
3391 if (!this->emitGetPtrLocal(LocalIndex, E))
3395 return this->
visit(
Init) && this->emitInit(*T, E);
3399template <
class Emitter>
3412template <
class Emitter>
3416 return this->emitConst(E->
getValue(), E);
3419template <
class Emitter>
3432 for (
const Record::Field &F : R->fields()) {
3434 if (!
Init ||
Init->containsErrors())
3442 if (!this->emitInitField(*T, F.Offset, E))
3445 if (!this->emitGetPtrField(F.Offset, E))
3456template <
class Emitter>
3463 return this->emitGetPtrGlobal(StringIndex, E);
3469template <
class Emitter>
3474 return this->emitInvalid(E);
3477template <
class Emitter>
3503 if (PointeeToT && PointeeFromT) {
3522 bool Fatal = (ToT != FromT);
3529template <
class Emitter>
3531 if (!
Ctx.getLangOpts().CPlusPlus20) {
3552 if (!this->emitDynamicCast(DestType.
getTypePtr(),
3557 return this->emitPopPtr(E);
3561template <
class Emitter>
3567 return this->emitConstBool(E->
getValue(), E);
3570template <
class Emitter>
3575 if (T->isRecordType()) {
3589 if (!this->emitGetPtrLocal(*LocalIndex, E))
3597 T->getAsCXXRecordDecl()))
3607 if (!this->visitZeroRecordInitializer(R, E))
3619 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3621 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3622 if (!this->emitCheckFunctionDecl(Ctor, E))
3633 assert(
Func->hasThisPointer());
3634 assert(!
Func->hasRVO());
3638 if (!this->emitDupPtr(E))
3642 for (
const auto *Arg : E->
arguments()) {
3643 if (!this->
visit(Arg))
3647 if (
Func->isVariadic()) {
3648 uint32_t VarArgSize = 0;
3649 unsigned NumParams =
Func->getNumWrittenParams();
3650 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3654 if (!this->emitCallVar(
Func, VarArgSize, E))
3657 if (!this->emitCall(
Func, 0, E)) {
3662 (void)this->emitPopPtr(E);
3668 return this->emitPopPtr(E);
3672 if (T->isArrayType()) {
3677 if (!this->emitDupPtr(E))
3681 initArrayDimension = [&](
QualType T) ->
bool {
3682 if (!T->isArrayType()) {
3684 for (
const auto *Arg : E->
arguments()) {
3685 if (!this->
visit(Arg))
3689 return this->emitCall(
Func, 0, E);
3693 Ctx.getASTContext().getAsConstantArrayType(T);
3698 for (
size_t I = 0; I != NumElems; ++I) {
3699 if (!this->emitConstUint64(I, E))
3701 if (!this->emitArrayElemPtrUint64(E))
3703 if (!initArrayDimension(ElemTy))
3706 return this->emitPopPtr(E);
3709 return initArrayDimension(E->
getType());
3715template <
class Emitter>
3725 assert(Val.
isInt());
3727 return this->emitConst(I, E);
3734 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3735 return this->
visit(LValueExpr);
3750 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3754 const APValue &
V = UGCD->getValue();
3755 for (
unsigned I = 0, N = R->getNumFields(); I != N; ++I) {
3756 const Record::Field *F = R->getField(I);
3757 const APValue &FieldValue =
V.getStructField(I);
3763 if (!this->emitInitField(FieldT, F->Offset, E))
3771template <
class Emitter>
3777 for (
unsigned I = 0; I != N; ++I) {
3784 if (!this->
discard(ArrayIndexExpr))
3789 if (!this->
visit(ArrayIndexExpr))
3793 if (!this->emitCast(IndexT,
PT_Sint64, E))
3803 return this->emitOffsetOf(T, E, E);
3806template <
class Emitter>
3815 return this->visitZeroInitializer(*T, Ty, E);
3822 if (!this->emitGetPtrLocal(*LocalIndex, E))
3830 ElemQT = CT->getElementType();
3833 NumElems = VT->getNumElements();
3834 ElemQT = VT->getElementType();
3840 for (
unsigned I = 0, N = NumElems; I != N; ++I) {
3841 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3843 if (!this->emitInitElem(ElemT, I, E))
3852template <
class Emitter>
3857template <
class Emitter>
3863template <
class Emitter>
3868template <
class Emitter>
3873 return this->emitConst(E->
getValue(), E);
3876template <
class Emitter>
3881 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3895 unsigned ParamIndex = 0;
3899 if (!this->emitGetParam(PT, ParamIndex, E))
3904 return this->emitCall(F, 0, E);
3909template <
class Emitter>
3917 const Expr *PlacementDest =
nullptr;
3918 bool IsNoThrow =
false;
3923 if (PlacementArgs != 0) {
3932 if (PlacementArgs == 1) {
3940 if (!this->emitInvalidNewDeleteExpr(E, E))
3945 if (OperatorNew->isReservedGlobalPlacementOperator())
3946 PlacementDest = Arg1;
3950 return this->emitInvalid(E);
3952 }
else if (!OperatorNew
3953 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3954 return this->emitInvalidNewDeleteExpr(E, E);
3957 if (!PlacementDest) {
3962 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3965 Desc =
P.createDescriptor(
3968 false,
false,
false,
3974 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3978 const Expr *Stripped = *ArraySizeExpr;
3979 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
3980 Stripped = ICE->getSubExpr())
3981 if (ICE->getCastKind() != CK_NoOp &&
3982 ICE->getCastKind() != CK_IntegralCast)
3990 if (!this->
visit(Stripped))
3992 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
3995 if (PlacementDest) {
3996 if (!this->
visit(PlacementDest))
3998 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4000 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
4003 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4008 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
4012 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
4019 size_t StaticInitElems = 0;
4020 const Expr *DynamicInit =
nullptr;
4024 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
4025 StaticInitElems = CAT->getZExtSize();
4030 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init)) {
4031 if (ILE->hasArrayFiller())
4032 DynamicInit = ILE->getArrayFiller();
4051 const Function *CtorFunc =
nullptr;
4052 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
4056 }
else if (!DynamicInit && !ElemT)
4059 LabelTy EndLabel = this->getLabel();
4060 LabelTy StartLabel = this->getLabel();
4065 if (!this->emitDupPtr(E))
4067 if (!this->emitNullPtr(0,
nullptr, E))
4069 if (!this->emitEQPtr(E))
4071 if (!this->jumpTrue(EndLabel, E))
4078 if (!this->emitConst(StaticInitElems,
SizeT, E))
4080 if (!this->emitSetLocal(
SizeT, Iter, E))
4083 this->fallthrough(StartLabel);
4084 this->emitLabel(StartLabel);
4086 if (!this->emitGetLocal(
SizeT, Iter, E))
4088 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4090 if (!this->emitLT(
SizeT, E))
4092 if (!this->jumpFalse(EndLabel, E))
4096 if (!this->emitGetLocal(
SizeT, Iter, E))
4098 if (!this->emitArrayElemPtr(
SizeT, E))
4101 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
4106 if (!this->visitZeroInitializer(*InitT, ElemType, E))
4108 if (!this->emitStorePop(*InitT, E))
4112 if (!this->visitZeroArrayInitializer(ElemType, E))
4115 }
else if (DynamicInit) {
4117 if (!this->
visit(DynamicInit))
4119 if (!this->emitStorePop(*InitT, E))
4126 if (!this->visitZeroInitializer(
4130 if (!this->emitStorePop(*ElemT, E))
4134 if (!this->emitCall(CtorFunc, 0, E))
4139 if (!this->emitGetPtrLocal(Iter, E))
4141 if (!this->emitIncPop(
SizeT,
false, E))
4144 if (!this->jump(StartLabel, E))
4147 this->fallthrough(EndLabel);
4148 this->emitLabel(EndLabel);
4152 if (PlacementDest) {
4153 if (!this->
visit(PlacementDest))
4155 if (!this->emitCheckNewTypeMismatch(E, E))
4160 if (!this->emitAlloc(Desc, E))
4169 if (!this->emitInit(*ElemT, E))
4180 return this->emitPopPtr(E);
4185template <
class Emitter>
4191 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
4192 return this->emitInvalidNewDeleteExpr(E, E);
4201template <
class Emitter>
4207 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
4212 return this->emitGetFnPtr(
Func, E);
4215template <
class Emitter>
4219 auto canonType = [](
const Type *T) {
4220 return T->getCanonicalTypeUnqualified().getTypePtr();
4228 return this->emitGetTypeid(
4232 return this->emitGetTypeid(
4241 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
4247 if (!this->emitGetTypeidPtr(TypeInfoType, E))
4250 return this->emitPopPtr(E);
4254template <
class Emitter>
4258 return this->emitDummyPtr(E, E);
4259 return this->emitError(E);
4262template <
class Emitter>
4265 return this->emitDummyPtr(E, E);
4266 return this->emitError(E);
4269template <
class Emitter>
4271 assert(
Ctx.getLangOpts().CPlusPlus);
4272 return this->emitConstBool(E->
getValue(), E);
4275template <
class Emitter>
4287 return this->emitDummyPtr(GuidDecl, E);
4292 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
4301 assert(
V.isStruct());
4302 assert(
V.getStructNumBases() == 0);
4306 return this->emitFinishInit(E);
4309template <
class Emitter>
4319template <
class Emitter>
4328template <
class Emitter>
4334template <
class Emitter>
4338 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
4342 if (OVE->isUnique())
4358template <
class Emitter>
4363template <
class Emitter>
4365 return this->emitError(E);
4368template <
class Emitter>
4374 return this->emitDummyPtr(E, E);
4377template <
class Emitter>
4378bool Compiler<Emitter>::emitVectorConversion(
const Expr *Src,
const Expr *E) {
4383 QualType ElemType = VT->getElementType();
4384 PrimType ElemT = classifyPrim(ElemType);
4386 PrimType SrcElemT = classifyVectorElementType(SrcType);
4392 if (!this->emitGetPtrLocal(*LocalIndex, E))
4396 unsigned SrcOffset =
4397 this->allocateLocalPrimitive(Src,
PT_Ptr,
true);
4398 if (!this->visit(Src))
4400 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4403 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4404 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4406 if (!this->emitArrayElemPop(SrcElemT, I, E))
4410 if (SrcElemT != ElemT) {
4411 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4413 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4414 const auto *TargetSemantics = &Ctx.getFloatSemantics(ElemType);
4418 if (!this->emitInitElem(ElemT, I, E))
4424template <
class Emitter>
4426 return emitVectorConversion(E->
getSrcExpr(), E);
4429template <
class Emitter>
4433 return this->emitInvalid(E);
4442 assert(NumOutputElems > 0);
4448 if (!this->emitGetPtrLocal(*LocalIndex, E))
4453 unsigned VectorOffsets[2];
4454 for (
unsigned I = 0; I != 2; ++I) {
4457 if (!this->
visit(Vecs[I]))
4459 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4462 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4464 assert(ShuffleIndex >= -1);
4465 if (ShuffleIndex == -1)
4466 return this->emitInvalidShuffleVectorIndex(I, E);
4468 assert(ShuffleIndex < (NumInputElems * 2));
4469 if (!this->emitGetLocal(
PT_Ptr,
4470 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4472 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4473 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4476 if (!this->emitInitElem(ElemT, I, E))
4481 return this->emitPopPtr(E);
4486template <
class Emitter>
4491 Base->getType()->isVectorType() ||
4497 if (Indices.size() == 1) {
4502 if (!this->emitConstUint32(Indices[0], E))
4504 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4514 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4522 if (!this->emitGetPtrLocal(*ResultIndex, E))
4530 uint32_t DstIndex = 0;
4531 for (uint32_t I : Indices) {
4532 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4534 if (!this->emitArrayElemPop(ElemT, I, E))
4536 if (!this->emitInitElem(ElemT, DstIndex, E))
4546template <
class Emitter>
4550 return this->
discard(SubExpr) && this->emitInvalid(E);
4556 return this->emitDummyPtr(E, E);
4559template <
class Emitter>
4564 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4569 if (!this->
visit(SubExpr))
4571 if (!this->emitConstUint8(0, E))
4573 if (!this->emitArrayElemPtrPopUint8(E))
4575 if (!this->emitInitFieldPtr(R->getField(0u)->Offset, E))
4580 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4582 return this->emitInitField(SecondFieldT, R->getField(1u)->Offset, E);
4584 assert(SecondFieldT ==
PT_Ptr);
4586 if (!this->emitGetFieldPtr(R->getField(0u)->Offset, E))
4588 if (!this->emitExpandPtr(E))
4592 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4594 return this->emitInitFieldPtr(R->getField(1u)->Offset, E);
4597template <
class Emitter>
4612 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4616 return this->emitUnsupported(E);
4625 return this->
Visit(E);
4632 return this->
Visit(E);
4636 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4639 if (
const auto *CE = dyn_cast<CastExpr>(E);
4641 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4648 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4651 if (
const auto *CE = dyn_cast<CastExpr>(E);
4652 CE && (CE->getCastKind() == CK_DerivedToBase ||
4653 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4654 CE->getCastKind() == CK_NoOp))
4674 if (!this->emitGetPtrLocal(*LocalIndex, E))
4684 return this->
Visit(E);
4687template <
class Emitter>
4693 return this->
Visit(E) && this->emitFinishInit(E);
4696template <
class Emitter>
4702 return this->
Visit(E) && this->emitFinishInitPop(E);
4708 return this->
Visit(E);
4719 if (!this->
visit(E))
4721 return this->emitComplexBoolCast(E);
4726 if (!this->
visit(E))
4734 return this->emitIsNonNullPtr(E);
4738 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4741 return this->emitCast(*T,
PT_Bool, E);
4744template <
class Emitter>
4748 QT = AT->getValueType();
4752 return this->emitZeroBool(E);
4754 return this->emitZeroSint8(E);
4756 return this->emitZeroUint8(E);
4758 return this->emitZeroSint16(E);
4760 return this->emitZeroUint16(E);
4762 return this->emitZeroSint32(E);
4764 return this->emitZeroUint32(E);
4766 return this->emitZeroSint64(E);
4768 return this->emitZeroUint64(E);
4770 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4772 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4774 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4777 return this->emitNullMemberPtr(0,
nullptr, E);
4779 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4780 return this->emitFloat(F, E);
4783 auto Sem = Ctx.getASTContext().getFixedPointSemantics(QT);
4787 llvm_unreachable(
"unknown primitive type");
4790template <
class Emitter>
4791bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4796 for (
const Record::Field &Field :
R->fields()) {
4797 if (Field.isUnnamedBitField())
4804 if (!this->visitZeroInitializer(T, QT, E))
4807 if (!this->emitInitFieldActivate(T, Field.Offset, E))
4811 if (!this->emitInitField(T, Field.Offset, E))
4816 if (!this->emitGetPtrField(Field.Offset, E))
4823 if (!this->visitZeroInitializer(T, ET, E))
4825 if (!this->emitInitElem(T, I, E))
4830 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4833 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4841 if (!this->emitFinishInitActivatePop(E))
4845 if (!this->emitFinishInitPop(E))
4849 for (
const Record::Base &B :
R->bases()) {
4850 if (!this->emitGetPtrBase(B.Offset, E))
4852 if (!this->visitZeroRecordInitializer(B.R, E))
4854 if (!this->emitFinishInitPop(E))
4863template <
class Emitter>
4864bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4865 assert(T->isArrayType() || T->isAnyComplexType() || T->isVectorType());
4871 for (
size_t I = 0; I != NumElems; ++I) {
4872 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4874 if (!this->emitInitElem(*ElemT, I, E))
4880 const Record *
R = getRecord(ElemType);
4884 for (
size_t I = 0; I != NumElems; ++I) {
4885 if (!this->emitConstUint32(I, E))
4887 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4889 if (!this->visitZeroRecordInitializer(R, E))
4891 if (!this->emitPopPtr(E))
4897 for (
size_t I = 0; I != NumElems; ++I) {
4898 if (!this->emitConstUint32(I, E))
4900 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4902 if (!this->visitZeroArrayInitializer(ElemType, E))
4904 if (!this->emitPopPtr(E))
4913template <
class Emitter>
4914bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4916 if (!canClassify(E->
getType()))
4919 if (!this->visit(RHS))
4921 if (!this->visit(LHS))
4928 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4932 bool Activates = refersToUnion(LHS);
4935 if (!this->emitFlip(
PT_Ptr, RHT, E))
4938 if (DiscardResult) {
4939 if (BitField && Activates)
4940 return this->emitStoreBitFieldActivatePop(RHT, E);
4942 return this->emitStoreBitFieldPop(RHT, E);
4944 return this->emitStoreActivatePop(RHT, E);
4946 return this->emitStorePop(RHT, E);
4949 auto maybeLoad = [&](
bool Result) ->
bool {
4955 return this->emitLoadPop(RHT, E);
4959 if (BitField && Activates)
4960 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4962 return maybeLoad(this->emitStoreBitField(RHT, E));
4964 return maybeLoad(this->emitStoreActivate(RHT, E));
4966 return maybeLoad(this->emitStore(RHT, E));
4969template <
class Emitter>
4970template <
typename T>
4974 return this->emitConstSint8(
Value, Info);
4976 return this->emitConstUint8(
Value, Info);
4978 return this->emitConstSint16(
Value, Info);
4980 return this->emitConstUint16(
Value, Info);
4982 return this->emitConstSint32(
Value, Info);
4984 return this->emitConstUint32(
Value, Info);
4986 return this->emitConstSint64(
Value, Info);
4988 return this->emitConstUint64(
Value, Info);
4990 return this->emitConstBool(
Value, Info);
4997 llvm_unreachable(
"Invalid integral type");
5000 llvm_unreachable(
"unknown primitive type");
5003template <
class Emitter>
5004template <
typename T>
5005bool Compiler<Emitter>::emitConst(T
Value,
const Expr *E) {
5006 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5009template <
class Emitter>
5013 return this->emitConstIntAPS(
Value, Info);
5015 return this->emitConstIntAP(
Value, Info);
5017 if (
Value.isSigned())
5018 return this->emitConst(
Value.getSExtValue(), Ty, Info);
5019 return this->emitConst(
Value.getZExtValue(), Ty, Info);
5022template <
class Emitter>
5026 return this->emitConstIntAPS(
Value, Info);
5028 return this->emitConstIntAP(
Value, Info);
5031 return this->emitConst(
Value.getSExtValue(), Ty, Info);
5032 return this->emitConst(
Value.getZExtValue(), Ty, Info);
5035template <
class Emitter>
5036bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
5037 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5040template <
class Emitter>
5053 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
5054 Locals.insert({VD, Local});
5055 VarScope->addForScopeKind(Local, SC);
5056 return Local.Offset;
5059template <
class Emitter>
5064 bool IsTemporary =
false;
5065 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
5068 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
5069 Init = VarD->getInit();
5071 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
5082 return std::nullopt;
5087 Locals.insert({Key, Local});
5088 VarScope->addForScopeKind(Local, SC);
5089 return Local.Offset;
5092template <
class Emitter>
5102 return std::nullopt;
5112 return Local.Offset;
5115template <
class Emitter>
5117 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
5118 return PT->getPointeeType()->getAsCanonical<RecordType>();
5124 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
5128template <
class Emitter>
5130 return P.getOrCreateRecord(RD);
5133template <
class Emitter>
5135 return Ctx.getOrCreateFunction(FD);
5138template <
class Emitter>
5142 auto maybeDestroyLocals = [&]() ->
bool {
5143 if (DestroyToplevelScope)
5144 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
5145 return this->emitCheckAllocations(E);
5152 return this->emitRetVoid(E) && maybeDestroyLocals();
5160 return this->emitRet(*T, E) && maybeDestroyLocals();
5168 if (!this->emitGetPtrLocal(*LocalOffset, E))
5176 return this->emitRetValue(E) && maybeDestroyLocals();
5179 return maybeDestroyLocals() &&
false;
5182template <
class Emitter>
5184 bool DestroyToplevelScope) {
5188 return this->
visitExpr(E, DestroyToplevelScope);
5191template <
class Emitter>
5203 if (
auto GlobalIndex =
P.getGlobal(VD)) {
5204 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5218template <
class Emitter>
5220 bool ConstantContext) {
5223 if (!ConstantContext) {
5237 auto GlobalIndex =
P.getGlobal(VD);
5238 assert(GlobalIndex);
5240 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
5243 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
5247 auto Local =
Locals.find(VD);
5248 assert(Local !=
Locals.end());
5250 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
5253 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
5263 auto GlobalIndex =
P.getGlobal(VD);
5264 assert(GlobalIndex);
5265 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5274 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
5277template <
class Emitter>
5288 if (!this->isActive())
5293 if (
Init &&
Init->isValueDependent())
5297 auto checkDecl = [&]() ->
bool {
5299 return !NeedsOp || this->emitCheckDecl(VD, VD);
5306 if (!
P.getGlobal(*GlobalIndex)->isInitialized())
5309 if (
P.isGlobalInitialized(*GlobalIndex))
5313 }
else if ((GlobalIndex =
5328 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
5331 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
5334 if (!this->emitStartInit(
Init))
5340 if (!this->emitEndInit(
Init))
5343 return this->emitFinishInitGlobal(
Init);
5353 if (!
Init ||
Init->getType()->isVoidType())
5362 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
5374 if (!this->emitCheckRefInit(
Init))
5378 return this->emitSetLocal(*VarT, Offset, VD);
5386 if (!this->emitGetPtrLocal(*Offset,
Init))
5394template <
class Emitter>
5415 Locals.insert({VD, Local});
5418 if (!this->emitGetPtrLocal(Local.Offset, VD))
5424 return this->emitDestructionPop(D, VD);
5427template <
class Emitter>
5433 return this->emitConst(Val.
getInt(), ValType, Info);
5436 return this->emitFloat(F, Info);
5441 if (!this->emitGetMemberPtr(MemberDecl, Info))
5447 if (!this->emitCopyMemberPtrPath(PathEntry, IsDerived, Info))
5453 return this->emitNullMemberPtr(0,
nullptr, Info);
5458 return this->emitNull(ValType, 0,
nullptr, Info);
5463 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
5464 return this->
visit(BaseExpr);
5469 QualType EntryType = VD->getType();
5470 for (
auto &Entry : Path) {
5472 uint64_t Index = Entry.getAsArrayIndex();
5475 if (!this->emitConst(Index,
PT_Uint64, Info))
5477 if (!this->emitArrayElemPtrPop(
PT_Uint64, Info))
5479 EntryType = ElemType;
5489 const Decl *BaseOrMember = Entry.getAsBaseOrMember().getPointer();
5490 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
5492 if (!this->emitGetPtrFieldPop(EntryOffset, Info))
5494 EntryType = FD->getType();
5498 if (!this->emitGetPtrBasePop(BaseOffset,
false, Info))
5500 EntryType =
Ctx.getASTContext().getCanonicalTagType(
Base);
5512template <
class Emitter>
5522 const Record::Field *RF = R->getField(I);
5523 QualType FieldType = RF->Decl->getType();
5529 if (!this->emitInitField(*PT, RF->Offset, Info))
5532 if (!this->emitGetPtrField(RF->Offset, Info))
5536 if (!this->emitFinishInitPop(Info))
5546 if (I >= R->getNumBases())
5551 const Record::Base *RB = R->getBase(I);
5552 QualType BaseType =
Ctx.getASTContext().getCanonicalTagType(RB->Decl);
5554 if (!this->emitGetPtrBase(RB->Offset, Info))
5558 if (!this->emitFinishInitPop(Info))
5573 const Record::Field *RF = R->getField(UnionField);
5574 QualType FieldType = RF->Decl->getType();
5579 if (RF->isBitField())
5580 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
5582 return this->emitInitFieldActivate(*PT, RF->Offset, Info);
5585 if (!this->emitGetPtrField(RF->Offset, Info))
5587 if (!this->emitActivate(Info))
5591 return this->emitPopPtr(Info);
5595 const auto *ArrType = T->getAsArrayTypeUnsafe();
5596 QualType ElemType = ArrType->getElementType();
5599 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5600 const APValue &Elem = A >= InitializedElems
5609 if (!this->emitInitElem(*ElemT, A, Info))
5612 if (!this->emitConstUint32(A, Info))
5614 if (!this->emitArrayElemPtrUint32(Info))
5618 if (!this->emitPopPtr(Info))
5629template <
class Emitter>
5631 unsigned BuiltinID) {
5632 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5637 return this->emitConst(0, E);
5640 if (!this->emitStartSpeculation(E))
5642 LabelTy EndLabel = this->getLabel();
5643 if (!this->speculate(E, EndLabel))
5645 if (!this->emitEndSpeculation(E))
5647 this->fallthrough(EndLabel);
5655 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5656 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5657 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5658 BuiltinID == Builtin::BI__builtin_function_start) {
5661 return this->emitDummyPtr(E, E);
5672 if (!this->emitGetPtrLocal(*LocalIndex, E))
5677 switch (BuiltinID) {
5678 case Builtin::BI__builtin_object_size:
5679 case Builtin::BI__builtin_dynamic_object_size: {
5683 if (!this->
visit(Arg0))
5694 case Builtin::BI__assume:
5695 case Builtin::BI__builtin_assume:
5698 case Builtin::BI__atomic_is_lock_free:
5699 case Builtin::BI__atomic_always_lock_free: {
5710 for (
const auto *Arg : E->
arguments()) {
5711 if (!this->
visit(Arg))
5717 if (!this->emitCallBI(E, BuiltinID, E))
5726template <
class Emitter>
5747 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5748 DD && DD->isTrivial()) {
5750 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5752 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5753 this->emitPopPtr(E);
5761 bool HasRVO = !
ReturnType->isVoidType() && !T;
5769 if (!this->emitGetPtrLocal(*LocalIndex, E))
5777 if (!this->emitGetPtrLocal(*LocalIndex, E))
5781 if (!this->emitDupPtr(E))
5788 bool IsAssignmentOperatorCall =
false;
5789 bool ActivateLHS =
false;
5790 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5791 OCE && OCE->isAssignmentOp()) {
5795 assert(Args.size() == 2);
5796 const CXXRecordDecl *LHSRecord = Args[0]->getType()->getAsCXXRecordDecl();
5798 IsAssignmentOperatorCall =
true;
5799 std::reverse(Args.begin(), Args.end());
5805 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5806 MD && MD->isStatic()) {
5810 Args.erase(Args.begin());
5814 bool Devirtualized =
false;
5817 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5825 if (!this->
visit(Callee))
5827 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5829 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5831 if (!this->emitGetMemberPtrBase(E))
5834 const auto *InstancePtr = MC->getImplicitObjectArgument();
5841 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5842 Devirtualized =
true;
5843 if (!this->
visit(Stripped))
5846 if (!this->
visit(InstancePtr))
5850 if (!this->
visit(InstancePtr))
5854 }
else if (
const auto *PD =
5855 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5856 if (!this->emitCheckPseudoDtor(E))
5864 return this->emitEndLifetimePop(E);
5865 }
else if (!FuncDecl) {
5869 if (!this->
visit(Callee))
5871 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5880 if (IsAssignmentOperatorCall) {
5881 assert(Args.size() == 2);
5884 if (!this->emitFlip(Arg2T, Arg1T, E))
5898 assert(HasRVO ==
Func->hasRVO());
5900 bool HasQualifier =
false;
5901 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5902 HasQualifier = ME->hasQualifier();
5904 bool IsVirtual =
false;
5905 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5906 IsVirtual = !Devirtualized && MD->isVirtual();
5911 if (IsVirtual && !HasQualifier) {
5912 uint32_t VarArgSize = 0;
5913 unsigned NumParams =
5914 Func->getNumWrittenParams() +
5916 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5919 if (!this->emitCallVirt(
Func, VarArgSize, E))
5921 }
else if (
Func->isVariadic()) {
5922 uint32_t VarArgSize = 0;
5923 unsigned NumParams =
5924 Func->getNumWrittenParams() +
5926 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5928 if (!this->emitCallVar(
Func, VarArgSize, E))
5931 if (!this->emitCall(
Func, 0, E))
5940 uint32_t ArgSize = 0;
5941 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5947 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5949 if (!this->emitGetMemberPtrDecl(E))
5952 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5955 if (!this->emitCallPtr(ArgSize, E, E))
5966template <
class Emitter>
5973template <
class Emitter>
5980template <
class Emitter>
5985 return this->emitConstBool(E->
getValue(), E);
5988template <
class Emitter>
5994 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5995 return this->emitNullPtr(Val,
nullptr, E);
5998template <
class Emitter>
6006 return this->emitZero(T, E);
6009template <
class Emitter>
6014 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
6015 if (this->LambdaThisCapture.Offset > 0) {
6016 if (this->LambdaThisCapture.IsPtr)
6017 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
6018 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
6027 return this->emitThis(E);
6042 unsigned StartIndex = 0;
6043 unsigned EndIndex = 0;
6045 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
6047 EndIndex = StartIndex;
6054 for (; StartIndex > 0; --StartIndex) {
6064 if (StartIndex == 0 && EndIndex == 0)
6067 assert(StartIndex < EndIndex);
6070 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
6082 case Stmt::CompoundStmtClass:
6084 case Stmt::DeclStmtClass:
6086 case Stmt::ReturnStmtClass:
6088 case Stmt::IfStmtClass:
6090 case Stmt::WhileStmtClass:
6092 case Stmt::DoStmtClass:
6094 case Stmt::ForStmtClass:
6096 case Stmt::CXXForRangeStmtClass:
6098 case Stmt::BreakStmtClass:
6100 case Stmt::ContinueStmtClass:
6102 case Stmt::SwitchStmtClass:
6104 case Stmt::CaseStmtClass:
6106 case Stmt::DefaultStmtClass:
6108 case Stmt::AttributedStmtClass:
6110 case Stmt::CXXTryStmtClass:
6112 case Stmt::NullStmtClass:
6115 case Stmt::GCCAsmStmtClass:
6116 case Stmt::MSAsmStmtClass:
6117 case Stmt::GotoStmtClass:
6118 return this->emitInvalid(S);
6119 case Stmt::LabelStmtClass:
6122 if (
const auto *E = dyn_cast<Expr>(S))
6129template <
class Emitter>
6132 for (
const auto *InnerStmt : S->
body())
6135 return Scope.destroyLocals();
6138template <
class Emitter>
6139bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
6140 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
6141 for (
auto *BD : DD->flat_bindings())
6142 if (
auto *KD = BD->getHoldingVar();
6143 KD && !this->visitVarDecl(KD, KD->getInit()))
6157template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
6159 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
6160 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
6167 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
6172 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
6173 ICE && (ICE->getCastKind() == CK_NoOp ||
6174 ICE->getCastKind() == CK_DerivedToBase ||
6175 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
6176 E = ICE->getSubExpr();
6180 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
6181 const auto *ThisRecord =
6182 This->getType()->getPointeeType()->getAsRecordDecl();
6183 if (!ThisRecord->isUnion())
6186 if (
const auto *Ctor =
6187 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
6188 return Ctor->getParent() == ThisRecord;
6197template <
class Emitter>
6199 bool EvaluateConditionDecl) {
6200 for (
const auto *D : DS->
decls()) {
6205 const auto *VD = dyn_cast<VarDecl>(D);
6212 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
6219template <
class Emitter>
6222 return this->emitUnsupported(RS);
6228 if (!this->
visit(RE))
6234 if (RE->getType()->isVoidType()) {
6235 if (!this->
visit(RE))
6238 if (RE->containsErrors())
6243 if (!this->emitRVOPtr(RE))
6249 return this->emitRetVoid(RS);
6255 return this->emitRetVoid(RS);
6261 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
6268 if (
auto *CondInit = IS->
getInit()) {
6284 return visitChildStmt(IS->
getThen());
6286 return visitChildStmt(Else);
6292 if (!this->emitIsConstantContext(IS))
6295 if (!this->emitIsConstantContext(IS))
6297 if (!this->emitInv(IS))
6311 LabelTy LabelElse = this->getLabel();
6312 LabelTy LabelEnd = this->getLabel();
6313 if (!this->jumpFalse(LabelElse, IS))
6315 if (!visitChildStmt(IS->
getThen()))
6317 if (!this->jump(LabelEnd, IS))
6319 this->emitLabel(LabelElse);
6320 if (!visitChildStmt(Else))
6322 this->emitLabel(LabelEnd);
6324 LabelTy LabelEnd = this->getLabel();
6325 if (!this->jumpFalse(LabelEnd, IS))
6327 if (!visitChildStmt(IS->
getThen()))
6329 this->emitLabel(LabelEnd);
6338template <
class Emitter>
6343 LabelTy CondLabel = this->getLabel();
6344 LabelTy EndLabel = this->getLabel();
6348 this->fallthrough(CondLabel);
6349 this->emitLabel(CondLabel);
6365 if (!this->jumpFalse(EndLabel, S))
6375 if (!this->jump(CondLabel, S))
6377 this->fallthrough(EndLabel);
6378 this->emitLabel(EndLabel);
6387 LabelTy StartLabel = this->getLabel();
6388 LabelTy EndLabel = this->getLabel();
6389 LabelTy CondLabel = this->getLabel();
6393 this->fallthrough(StartLabel);
6394 this->emitLabel(StartLabel);
6400 this->fallthrough(CondLabel);
6401 this->emitLabel(CondLabel);
6408 if (!this->jumpTrue(StartLabel, S))
6411 this->fallthrough(EndLabel);
6412 this->emitLabel(EndLabel);
6416template <
class Emitter>
6424 LabelTy EndLabel = this->getLabel();
6425 LabelTy CondLabel = this->getLabel();
6426 LabelTy IncLabel = this->getLabel();
6433 this->fallthrough(CondLabel);
6434 this->emitLabel(CondLabel);
6446 if (!this->jumpFalse(EndLabel, S))
6455 this->fallthrough(IncLabel);
6456 this->emitLabel(IncLabel);
6462 if (!this->jump(CondLabel, S))
6466 this->emitLabel(EndLabel);
6472template <
class Emitter>
6482 LabelTy EndLabel = this->getLabel();
6483 LabelTy CondLabel = this->getLabel();
6484 LabelTy IncLabel = this->getLabel();
6499 this->fallthrough(CondLabel);
6500 this->emitLabel(CondLabel);
6503 if (!this->jumpFalse(EndLabel, S))
6514 this->fallthrough(IncLabel);
6515 this->emitLabel(IncLabel);
6520 if (!this->jump(CondLabel, S))
6523 this->fallthrough(EndLabel);
6524 this->emitLabel(EndLabel);
6528template <
class Emitter>
6539 if (LI.BreakLabel) {
6540 TargetLabel = *LI.BreakLabel;
6541 BreakScope = LI.BreakOrContinueScope;
6547 if (LI.Name == TargetLoop) {
6548 TargetLabel = *LI.BreakLabel;
6549 BreakScope = LI.BreakOrContinueScope;
6560 C =
C->getParent()) {
6561 if (!
C->destroyLocals())
6565 return this->jump(*TargetLabel, S);
6568template <
class Emitter>
6579 if (LI.ContinueLabel) {
6580 TargetLabel = *LI.ContinueLabel;
6581 ContinueScope = LI.BreakOrContinueScope;
6587 if (LI.Name == TargetLoop) {
6588 TargetLabel = *LI.ContinueLabel;
6589 ContinueScope = LI.BreakOrContinueScope;
6594 assert(TargetLabel);
6597 C =
C->getParent()) {
6598 if (!
C->destroyLocals())
6602 return this->jump(*TargetLabel, S);
6605template <
class Emitter>
6608 if (
Cond->containsErrors())
6615 LabelTy EndLabel = this->getLabel();
6620 if (
const auto *CondInit = S->
getInit())
6631 if (!this->emitSetLocal(CondT, CondVar, S))
6641 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6644 if (CS->caseStmtIsGNURange()) {
6645 LabelTy EndOfRangeCheck = this->getLabel();
6646 const Expr *Low = CS->getLHS();
6647 const Expr *High = CS->getRHS();
6651 if (!this->emitGetLocal(CondT, CondVar, CS))
6653 if (!this->
visit(Low))
6656 if (!this->emitGE(
LT, S))
6658 if (!this->jumpFalse(EndOfRangeCheck, S))
6661 if (!this->emitGetLocal(CondT, CondVar, CS))
6663 if (!this->
visit(High))
6666 if (!this->emitLE(HT, S))
6670 this->emitLabel(EndOfRangeCheck);
6675 if (
Value->isValueDependent())
6680 if (!this->emitGetLocal(CondT, CondVar, CS))
6686 if (!this->emitEQ(ValueT, S))
6691 assert(!DefaultLabel);
6692 DefaultLabel = this->getLabel();
6699 if (!this->jump(*DefaultLabel, S))
6702 if (!this->jump(EndLabel, S))
6710 this->fallthrough(EndLabel);
6711 this->emitLabel(EndLabel);
6716template <
class Emitter>
6724 return this->emitUnsupported(S);
6729template <
class Emitter>
6736 if (LI.DefaultLabel) {
6737 DefaultLabel = *LI.DefaultLabel;
6742 this->emitLabel(DefaultLabel);
6746template <
class Emitter>
6753 if (IsMSVCConstexprAttr && !this->emitPushMSVCCE(S))
6756 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6757 !this->Ctx.getLangOpts().MSVCCompat) {
6759 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6765 const Expr *Assumption = AA->getAssumption();
6776 if (!this->emitAssume(Assumption))
6785 if (IsMSVCConstexprAttr)
6786 return this->emitPopMSVCCE(S);
6790template <
class Emitter>
6796template <
class Emitter>
6797bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6804 assert(ClosureClass->
captures().empty());
6808 "A generic lambda's static-invoker function must be a "
6809 "template specialization");
6813 void *InsertPos =
nullptr;
6814 const FunctionDecl *CorrespondingCallOpSpecialization =
6816 assert(CorrespondingCallOpSpecialization);
6817 LambdaCallOp = CorrespondingCallOpSpecialization;
6821 assert(ClosureClass->
captures().empty());
6822 const Function *
Func = this->getFunction(LambdaCallOp);
6825 assert(
Func->hasThisPointer());
6828 if (
Func->hasRVO()) {
6829 if (!this->emitRVOPtr(MD))
6837 if (!this->emitNullPtr(0,
nullptr, MD))
6842 auto It = this->Params.find(PVD);
6843 assert(It != this->Params.end());
6847 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6848 if (!this->emitGetParam(ParamType, It->second.Index, MD))
6852 if (!this->emitCall(
Func, 0, LambdaCallOp))
6857 return this->emitRet(*ReturnType, MD);
6860 return this->emitRetVoid(MD);
6863template <
class Emitter>
6864bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6865 if (Ctx.getLangOpts().CPlusPlus23)
6875 const Expr *InitExpr =
Init->getInit();
6877 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6881 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6891template <
class Emitter>
6893 assert(!ReturnType);
6896 if (!this->emitStartThisLifetime1(Ctor))
6899 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6900 const Expr *InitExpr,
6903 if (InitExpr->getType().isNull())
6907 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6910 if (!this->visit(InitExpr))
6913 if (F->isBitField())
6914 return this->emitInitThisBitField(*T, FieldOffset, F->bitWidth(),
6916 return this->emitInitThisField(*T, FieldOffset, InitExpr);
6921 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6924 if (
Activate && !this->emitActivate(InitExpr))
6927 return this->visitInitializerPop(InitExpr);
6931 const Record *
R = this->getRecord(RD);
6934 bool IsUnion =
R->isUnion();
6944 if (!this->emitThis(Ctor))
6947 if (!this->emitGetParam(
PT_Ptr, 0, Ctor))
6950 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6951 this->emitRetVoid(Ctor);
6954 unsigned FieldInits = 0;
6956 for (
const auto *
Init : Ctor->
inits()) {
6960 const Expr *InitExpr =
Init->getInit();
6962 const Record::Field *F =
R->getField(
Member);
6966 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6970 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6973 if (
Init->isBaseVirtual()) {
6974 assert(
R->getVirtualBase(BaseDecl));
6975 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6981 const Record::Base *B =
R->getBase(BaseDecl);
6983 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6987 if (!this->visitInitializerPop(InitExpr))
6992 unsigned ChainSize = IFD->getChainingSize();
6993 assert(ChainSize >= 2);
6995 unsigned NestedFieldOffset = 0;
6996 const Record::Field *NestedField =
nullptr;
6997 for (
unsigned I = 0; I != ChainSize; ++I) {
6999 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
7000 assert(FieldRecord);
7002 NestedField = FieldRecord->
getField(FD);
7003 assert(NestedField);
7004 IsUnion = IsUnion || FieldRecord->
isUnion();
7006 NestedFieldOffset += NestedField->Offset;
7009 if (I != ChainSize - 1)
7012 assert(NestedField);
7015 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
7020 unsigned InitFieldOffset = 0;
7021 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
7023 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
7024 assert(FieldRecord);
7025 NestedField = FieldRecord->
getField(FD);
7026 InitFieldOffset += NestedField->Offset;
7027 assert(NestedField);
7028 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
7030 if (!this->emitFinishInitPop(InitExpr))
7034 InitStack.pop_back_n(ChainSize - 1);
7037 assert(
Init->isDelegatingInitializer());
7038 if (!this->emitThis(InitExpr))
7040 if (!this->visitInitializerPop(
Init->getInit()))
7044 if (!
Scope.destroyLocals())
7048 if (FieldInits !=
R->getNumFields()) {
7049 assert(FieldInits < R->getNumFields());
7051 if (!this->emitStartThisLifetime(Ctor))
7058 if (
const auto *CS = dyn_cast<CompoundStmt>(Body)) {
7059 if (!CS->body_empty() && !this->emitCtorCheck(
SourceInfo{}))
7066 if (!visitStmt(Body))
7073template <
class Emitter>
7076 const Record *
R = this->getRecord(RD);
7080 if (!
Dtor->isTrivial() &&
Dtor->getBody()) {
7081 if (!this->visitStmt(
Dtor->getBody()))
7085 if (!this->emitThis(
Dtor))
7088 if (!this->emitCheckDestruction(
Dtor))
7092 if (!
R->isUnion()) {
7096 for (
const Record::Field &Field : llvm::reverse(
R->fields())) {
7100 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
7102 if (!this->emitDestructionPop(D,
SourceInfo{}))
7107 for (
const Record::Base &
Base : llvm::reverse(
R->bases())) {
7108 if (
Base.R->hasTrivialDtor())
7112 if (!this->emitRecordDestructionPop(
Base.R, {}))
7116 if (!this->emitMarkDestroyed(
Dtor))
7120 return this->emitPopPtr(
Dtor) && this->emitRetVoid(
Dtor);
7123template <
class Emitter>
7124bool Compiler<Emitter>::compileUnionAssignmentOperator(
7126 if (!this->emitThis(MD))
7129 if (!this->emitGetParam(
PT_Ptr, 0, MD))
7132 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
7135template <
class Emitter>
7145 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
7146 return this->compileConstructor(Ctor);
7147 if (
const auto *
Dtor = dyn_cast<CXXDestructorDecl>(F))
7148 return this->compileDestructor(
Dtor);
7151 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
7156 return this->compileUnionAssignmentOperator(MD);
7159 return this->emitLambdaStaticInvokerBody(MD);
7163 if (
const auto *Body = F->
getBody())
7177 return FD->getBitWidthValue();
7180template <
class Emitter>
7196 if (!
Ctx.getLangOpts().CPlusPlus14)
7197 return this->emitInvalid(E);
7199 return this->emitError(E);
7201 if (!this->
visit(SubExpr))
7205 if (!this->emitIncPtr(E))
7212 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
7213 : this->emitIncf(getFPOptions(E), E);
7225 if (!
Ctx.getLangOpts().CPlusPlus14)
7226 return this->emitInvalid(E);
7228 return this->emitError(E);
7230 if (!this->
visit(SubExpr))
7234 if (!this->emitDecPtr(E))
7241 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
7242 : this->emitDecf(getFPOptions(E), E);
7255 if (!
Ctx.getLangOpts().CPlusPlus14)
7256 return this->emitInvalid(E);
7258 return this->emitError(E);
7260 if (!this->
visit(SubExpr))
7264 if (!this->emitLoadPtr(E))
7266 if (!this->emitConstUint8(1, E))
7268 if (!this->emitAddOffsetUint8(E))
7270 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7276 return this->emitIncfPop(getFPOptions(E), E);
7286 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7287 if (!this->emitLoadFloat(E))
7289 APFloat F(TargetSemantics, 1);
7290 if (!this->emitFloat(F, E))
7293 if (!this->emitAddf(getFPOptions(E), E))
7295 if (!this->emitStoreFloat(E))
7307 return E->
isGLValue() || this->emitLoadPop(*T, E);
7310 if (!
Ctx.getLangOpts().CPlusPlus14)
7311 return this->emitInvalid(E);
7313 return this->emitError(E);
7315 if (!this->
visit(SubExpr))
7319 if (!this->emitLoadPtr(E))
7321 if (!this->emitConstUint8(1, E))
7323 if (!this->emitSubOffsetUint8(E))
7325 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7331 return this->emitDecfPop(getFPOptions(E), E);
7341 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7342 if (!this->emitLoadFloat(E))
7344 APFloat F(TargetSemantics, 1);
7345 if (!this->emitFloat(F, E))
7348 if (!this->emitSubf(getFPOptions(E), E))
7350 if (!this->emitStoreFloat(E))
7362 return E->
isGLValue() || this->emitLoadPop(*T, E);
7366 return this->emitError(E);
7369 return this->
discard(SubExpr);
7374 if (!this->emitInv(E))
7378 return this->emitCast(
PT_Bool, ET, E);
7382 return this->emitError(E);
7384 if (!this->
visit(SubExpr))
7386 return DiscardResult ? this->emitPop(*T, E) : this->emitNeg(*T, E);
7389 return this->emitError(E);
7391 if (!this->
visit(SubExpr))
7407 if (!
Ctx.getLangOpts().CPlusPlus) {
7409 if (
const auto *Deref = dyn_cast<UnaryOperator>(
Sub);
7410 Deref && Deref->getOpcode() == UO_Deref)
7411 return this->
delegate(Deref->getSubExpr());
7417 return this->
discard(SubExpr);
7419 if (!this->
visit(SubExpr))
7426 return this->emitNarrowPtr(E);
7431 return this->emitError(E);
7433 if (!this->
visit(SubExpr))
7435 return DiscardResult ? this->emitPop(*T, E) : this->emitComp(*T, E);
7447 : this->visitZeroInitializer(*T, SubExpr->
getType(), SubExpr);
7452 assert(
false &&
"Unhandled opcode");
7458template <
class Emitter>
7464 return this->
discard(SubExpr);
7467 auto prepareResult = [=]() ->
bool {
7472 return this->emitGetPtrLocal(*LocalIndex, E);
7479 unsigned SubExprOffset = ~0u;
7480 auto createTemp = [=, &SubExprOffset]() ->
bool {
7483 if (!this->
visit(SubExpr))
7485 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
7489 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7490 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7492 return this->emitArrayElemPop(ElemT, Index, E);
7497 if (!prepareResult())
7501 for (
unsigned I = 0; I != 2; ++I) {
7502 if (!getElem(SubExprOffset, I))
7504 if (!this->emitNeg(ElemT, E))
7506 if (!this->emitInitElem(ElemT, I, E))
7517 if (!this->
visit(SubExpr))
7519 if (!this->emitComplexBoolCast(SubExpr))
7521 if (!this->emitInv(E))
7524 return this->emitCast(
PT_Bool, ET, E);
7528 return this->emitComplexReal(SubExpr);
7531 if (!this->
visit(SubExpr))
7535 if (!this->emitConstUint8(1, E))
7537 return this->emitArrayElemPtrPopUint8(E);
7548 if (!this->emitArrayElem(ElemT, 1, E))
7550 if (!this->emitNeg(ElemT, E))
7552 if (!this->emitInitElem(ElemT, 1, E))
7560 return this->emitInvalid(E);
7566template <
class Emitter>
7572 return this->
discard(SubExpr);
7575 if (UnaryOp == UO_Extension)
7578 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
7579 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
7580 return this->emitInvalid(E);
7583 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
7590 if (!this->emitGetPtrLocal(*LocalIndex, E))
7595 unsigned SubExprOffset =
7597 if (!this->
visit(SubExpr))
7599 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
7604 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7605 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7607 return this->emitArrayElemPop(ElemT, Index, E);
7612 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7613 if (!getElem(SubExprOffset, I))
7615 if (!this->emitNeg(ElemT, E))
7617 if (!this->emitInitElem(ElemT, I, E))
7632 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7633 if (!getElem(SubExprOffset, I))
7636 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
7638 if (!this->emitInv(E))
7640 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
7642 if (!this->emitNeg(ElemT, E))
7644 if (ElemT != ResultVecElemT &&
7645 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
7647 if (!this->emitInitElem(ResultVecElemT, I, E))
7653 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7654 if (!getElem(SubExprOffset, I))
7657 if (!this->emitInv(E))
7660 if (!this->emitComp(ElemT, E))
7663 if (!this->emitInitElem(ElemT, I, E))
7668 llvm_unreachable(
"Unsupported unary operators should be handled up front");
7673template <
class Emitter>
7675 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D)) {
7678 return this->emitConst(ECD->getInitVal(), E);
7680 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
7684 return F && this->emitGetFnPtr(F, E);
7686 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
7694 return this->emitInitGlobal(*T, *Index, E);
7697 if (!this->emitGetPtrGlobal(*Index, E))
7701 return this->emitFinishInit(E);
7713 auto maybePopPtr = [&]() ->
bool {
7715 return this->emitPopPtr(E);
7722 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
7726 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7731 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7732 if (IsReference || !It->second.IsPtr)
7733 return this->emitGetParam(
classifyPrim(E), It->second.Index, E);
7735 return this->emitGetPtrParam(It->second.Index, E);
7738 if (!
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7744 const unsigned Offset = It->second.Offset;
7747 return this->emitGetRefLocal(Offset, E) && maybePopPtr();
7749 return this->emitGetPtrLocal(Offset, E) && maybePopPtr();
7752 if (
auto GlobalIndex =
P.getGlobal(D)) {
7754 if (!
Ctx.getLangOpts().CPlusPlus11)
7755 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7756 if (!
Ctx.getLangOpts().CPlusPlus23)
7757 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7759 return this->emitGetRefGlobal(*GlobalIndex, E) && maybePopPtr();
7762 return this->emitGetPtrGlobal(*GlobalIndex, E) && maybePopPtr();
7766 auto revisit = [&](
const VarDecl *VD,
7767 bool IsConstexprUnknown =
true) ->
bool {
7769 IsConstexprUnknown);
7774 if (!this->emitPopCC(E))
7777 if (VarState.notCreated())
7785 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
7787 if (
auto It = this->LambdaCaptures.find(D);
7788 It != this->LambdaCaptures.end()) {
7789 auto [Offset, IsPtr] = It->second;
7792 return this->emitGetThisFieldPtr(Offset, E) && maybePopPtr();
7793 return this->emitGetPtrThisField(Offset, E) && maybePopPtr();
7797 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7798 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7799 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7803 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7804 return this->
delegate(BD->getBinding());
7810 return this->emitDummyPtr(D, E);
7815 const auto *VD = dyn_cast<VarDecl>(D);
7817 return this->emitError(E);
7820 if (!
Ctx.getLangOpts().CPlusPlus) {
7824 return revisit(VD,
false);
7828 return this->emitDummyPtr(D, E);
7832 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7833 if (T.isConstant(
Ctx.getASTContext()))
7835 return T->isReferenceType();
7839 typeShouldBeVisited(DeclType)) {
7841 Init && !
Init->isValueDependent()) {
7846 (void)
Init->EvaluateAsInitializer(
Ctx.getASTContext(), VD,
Result,
true);
7859 bool IsConstexprUnknown = !DeclType.
isConstant(
Ctx.getASTContext()) &&
7864 return revisit(VD, IsConstexprUnknown);
7865 }
else if (
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7866 return revisit(VD,
true);
7875 return this->emitDummyPtr(
7879template <
class Emitter>
7885template <
class Emitter>
7895 if (!
C->destroyLocals())
7901template <
class Emitter>
7902unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7905 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7907 return Ty->getAsCXXRecordDecl();
7909 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7910 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7912 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7916template <
class Emitter>
7923 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7928 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7929 getFPOptions(E), E);
7931 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7932 getFPOptions(E), E);
7936 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7941 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7943 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7947 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7951 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7952 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7959template <
class Emitter>
7962 assert(FromT != ToT);
7965 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7967 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7969 return this->emitCast(FromT, ToT, E);
7973template <
class Emitter>
7974bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7978 return this->
discard(SubExpr);
7980 if (!this->visit(SubExpr))
7983 if (!this->emitConstUint8(0, SubExpr))
7985 return this->emitArrayElemPtrPopUint8(SubExpr);
7989 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7993template <
class Emitter>
7994bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7995 assert(!DiscardResult);
7999 if (!this->emitArrayElem(ElemT, 0, E))
8002 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
8005 if (!this->emitCast(ElemT,
PT_Bool, E))
8010 LabelTy LabelTrue = this->getLabel();
8011 if (!this->jumpTrue(LabelTrue, E))
8014 if (!this->emitArrayElemPop(ElemT, 1, E))
8017 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
8020 if (!this->emitCast(ElemT,
PT_Bool, E))
8024 LabelTy EndLabel = this->getLabel();
8025 this->jump(EndLabel, E);
8027 this->emitLabel(LabelTrue);
8028 if (!this->emitPopPtr(E))
8030 if (!this->emitConstBool(
true, E))
8033 this->fallthrough(EndLabel);
8034 this->emitLabel(EndLabel);
8039template <
class Emitter>
8040bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
8051 LHSIsComplex =
true;
8052 ElemT = classifyComplexElementType(LHS->
getType());
8053 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
8054 if (!this->visit(LHS))
8056 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
8059 LHSIsComplex =
false;
8061 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
8062 if (!this->visit(LHS))
8064 if (!this->emitSetLocal(LHST, LHSOffset, E))
8071 RHSIsComplex =
true;
8072 ElemT = classifyComplexElementType(RHS->
getType());
8073 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
8074 if (!this->visit(RHS))
8076 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
8079 RHSIsComplex =
false;
8081 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
8082 if (!this->visit(RHS))
8084 if (!this->emitSetLocal(RHST, RHSOffset, E))
8088 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
8089 bool IsComplex) ->
bool {
8091 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
8093 return this->emitArrayElemPop(ElemT, Index, E);
8095 return this->emitGetLocal(ElemT, LocalOffset, E);
8098 for (
unsigned I = 0; I != 2; ++I) {
8100 if (!getElem(LHSOffset, I, LHSIsComplex))
8102 if (!getElem(RHSOffset, I, RHSIsComplex))
8105 if (!this->emitEQ(ElemT, E))
8108 if (!this->emitCastBoolUint8(E))
8113 if (!this->emitAddUint8(E))
8115 if (!this->emitConstUint8(2, E))
8119 if (!this->emitEQUint8(E))
8122 if (!this->emitNEUint8(E))
8129 return this->emitCast(
PT_Bool, ResT, E);
8136template <
class Emitter>
8137bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
8140 assert(!
R->hasTrivialDtor());
8143 const Function *DtorFunc = getFunction(
Dtor);
8146 assert(DtorFunc->hasThisPointer());
8147 assert(DtorFunc->getNumParams() == 1);
8148 return this->emitCall(DtorFunc, 0, Loc);
8153template <
class Emitter>
8154bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
8166 return this->emitPopPtr(Loc);
8168 for (ssize_t I = N - 1; I >= 1; --I) {
8169 if (!this->emitConstUint64(I, Loc))
8171 if (!this->emitArrayElemPtrUint64(Loc))
8173 if (!this->emitDestructionPop(ElemDesc, Loc))
8177 if (!this->emitConstUint64(0, Loc))
8179 if (!this->emitArrayElemPtrPopUint64(Loc))
8181 return this->emitDestructionPop(ElemDesc, Loc);
8186 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
8191template <
class Emitter>
8192bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E,
bool CU) {
8193 assert(!DiscardResult &&
"Should've been checked before");
8194 unsigned DummyID = P.getOrCreateDummy(D, CU);
8196 if (!this->emitGetPtrGlobal(DummyID, E))
8204 return this->emitDecayPtr(
PT_Ptr, PT, E);
8210template <
class Emitter>
8213 return this->emitConstFloat(
Floating(F), Info);
8215 APInt I = F.bitcastToAPInt();
8216 return this->emitConstFloat(
8217 Floating(
const_cast<uint64_t *
>(I.getRawData()),
8218 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
8229template <
class Emitter>
8230bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
8243 if (!this->emitGetPtrLocal(*LocalIndex, E))
8253 if (!this->visit(SubExpr))
8255 }
else if (
OptPrimType FromT = classify(SubExpr)) {
8256 unsigned TempOffset =
8257 allocateLocalPrimitive(SubExpr, *FromT,
true);
8258 if (!this->visit(SubExpr))
8260 if (!this->emitSetLocal(*FromT, TempOffset, E))
8262 if (!this->emitGetPtrLocal(TempOffset, E))
8269 if (!this->emitBitCast(E))
8271 return DiscardResult ? this->emitPopPtr(E) :
true;
8275 const llvm::fltSemantics *TargetSemantics =
nullptr;
8277 TargetSemantics = &Ctx.getFloatSemantics(ToType);
8283 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
8285 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
8286 ResultBitWidth, TargetSemantics,
8291 return this->emitPop(*ToT, E);
8300template <
class Emitter>
8301bool Compiler<Emitter>::emitHLSLAggregateSplat(
PrimType SrcT,
8306 unsigned NumElems = 0;
8309 NumElems = VT->getNumElements();
8310 ElemType = VT->getElementType();
8312 NumElems = MT->getNumElementsFlattened();
8313 ElemType = MT->getElementType();
8316 PrimType ElemT = classifyPrim(ElemType);
8317 for (
unsigned I = 0; I != NumElems; ++I) {
8318 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8320 if (!this->emitPrimCast(SrcT, ElemT, ElemType, E))
8322 if (!this->emitInitElem(ElemT, I, E))
8332 QualType ArrElemType = CAT->getElementType();
8333 unsigned ArrSize = CAT->getZExtSize();
8336 for (
unsigned I = 0; I != ArrSize; ++I) {
8337 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8339 if (!this->emitPrimCast(SrcT, *ElemT, ArrElemType, E))
8341 if (!this->emitInitElem(*ElemT, I, E))
8345 for (
unsigned I = 0; I != ArrSize; ++I) {
8346 if (!this->emitConstUint32(I, E))
8348 if (!this->emitArrayElemPtrUint32(E))
8350 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, ArrElemType, E))
8352 if (!this->emitFinishInitPop(E))
8362 const Record *
R = getRecord(DestType);
8366 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8368 const Record::Base *B =
R->getBase(BS.getType());
8370 if (!this->emitGetPtrBase(B->Offset, E))
8372 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, BS.getType(), E))
8374 if (!this->emitFinishInitPop(E))
8379 for (
const Record::Field &F :
R->fields()) {
8380 if (F.isUnnamedBitField())
8383 QualType FieldType = F.Decl->getType();
8385 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8387 if (!this->emitPrimCast(SrcT, *FieldT, FieldType, E))
8389 if (F.isBitField()) {
8390 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8393 if (!this->emitInitField(*FieldT, F.Offset, E))
8397 if (!this->emitGetPtrField(F.Offset, E))
8399 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, FieldType, E))
8401 if (!this->emitPopPtr(E))
8414template <
class Emitter>
8415unsigned Compiler<Emitter>::countHLSLFlatElements(
QualType Ty) {
8418 return VT->getNumElements();
8420 return MT->getNumElementsFlattened();
8424 return CAT->getZExtSize() * countHLSLFlatElements(CAT->getElementType());
8428 const Record *
R = getRecord(Ty);
8432 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8434 Count += countHLSLFlatElements(BS.getType());
8436 for (
const Record::Field &F :
R->fields()) {
8437 if (F.isUnnamedBitField())
8439 Count += countHLSLFlatElements(F.Decl->getType());
8444 if (canClassify(Ty))
8453template <
class Emitter>
8454bool Compiler<Emitter>::emitHLSLFlattenAggregate(
8455 QualType SrcType,
unsigned SrcOffset,
8460 auto saveToLocal = [&](
PrimType T) ->
bool {
8461 unsigned Offset = allocateLocalPrimitive(E, T,
true);
8462 if (!this->emitSetLocal(T, Offset, E))
8464 Elements.push_back({Offset, T});
8470 unsigned Offset = allocateLocalPrimitive(E,
PT_Ptr,
true);
8471 if (!this->emitSetLocal(
PT_Ptr, Offset, E))
8472 return std::nullopt;
8477 unsigned NumElems = 0;
8480 NumElems = VT->getNumElements();
8481 ElemType = VT->getElementType();
8483 NumElems = MT->getNumElementsFlattened();
8484 ElemType = MT->getElementType();
8487 PrimType ElemT = classifyPrim(ElemType);
8488 for (
unsigned I = 0; I != NumElems && Elements.size() < MaxElements; ++I) {
8489 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8491 if (!this->emitArrayElemPop(ElemT, I, E))
8493 if (!saveToLocal(ElemT))
8503 QualType ArrElemType = CAT->getElementType();
8504 unsigned ArrSize = CAT->getZExtSize();
8507 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8508 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8510 if (!this->emitArrayElemPop(*ElemT, I, E))
8512 if (!saveToLocal(*ElemT))
8516 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8517 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8519 if (!this->emitConstUint32(I, E))
8521 if (!this->emitArrayElemPtrPopUint32(E))
8526 if (!emitHLSLFlattenAggregate(ArrElemType, *ElemPtrOffset, Elements,
8537 const Record *
R = getRecord(SrcType);
8541 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8543 if (Elements.size() >= MaxElements)
8545 const Record::Base *B =
R->getBase(BS.getType());
8547 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8549 if (!this->emitGetPtrBasePop(B->Offset,
false, E))
8554 if (!emitHLSLFlattenAggregate(BS.getType(), *BasePtrOffset, Elements,
8560 for (
const Record::Field &F :
R->fields()) {
8561 if (Elements.size() >= MaxElements)
8563 if (F.isUnnamedBitField())
8566 QualType FieldType = F.Decl->getType();
8567 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8569 if (!this->emitGetPtrFieldPop(F.Offset, E))
8573 if (!this->emitLoadPop(*FieldT, E))
8575 if (!saveToLocal(*FieldT))
8579 if (!FieldPtrOffset)
8581 if (!emitHLSLFlattenAggregate(FieldType, *FieldPtrOffset, Elements,
8597template <
class Emitter>
8598bool Compiler<Emitter>::emitHLSLConstructAggregate(
8604 const auto &Src = Elements[ElemIdx++];
8605 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
8607 return this->emitPrimCast(Src.Type, DestT, DestQT, E);
8611 unsigned NumElems = 0;
8614 NumElems = VT->getNumElements();
8615 ElemType = VT->getElementType();
8617 NumElems = MT->getNumElementsFlattened();
8618 ElemType = MT->getElementType();
8621 PrimType DestElemT = classifyPrim(ElemType);
8622 for (
unsigned I = 0; I != NumElems; ++I) {
8623 if (!loadAndCast(DestElemT, ElemType))
8625 if (!this->emitInitElem(DestElemT, I, E))
8635 QualType ArrElemType = CAT->getElementType();
8636 unsigned ArrSize = CAT->getZExtSize();
8639 for (
unsigned I = 0; I != ArrSize; ++I) {
8640 if (!loadAndCast(*ElemT, ArrElemType))
8642 if (!this->emitInitElem(*ElemT, I, E))
8646 for (
unsigned I = 0; I != ArrSize; ++I) {
8647 if (!this->emitConstUint32(I, E))
8649 if (!this->emitArrayElemPtrUint32(E))
8651 if (!emitHLSLConstructAggregate(ArrElemType, Elements, ElemIdx, E))
8653 if (!this->emitFinishInitPop(E))
8663 const Record *
R = getRecord(DestType);
8667 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8669 const Record::Base *B =
R->getBase(BS.getType());
8671 if (!this->emitGetPtrBase(B->Offset, E))
8673 if (!emitHLSLConstructAggregate(BS.getType(), Elements, ElemIdx, E))
8675 if (!this->emitFinishInitPop(E))
8680 for (
const Record::Field &F :
R->fields()) {
8681 if (F.isUnnamedBitField())
8684 QualType FieldType = F.Decl->getType();
8686 if (!loadAndCast(*FieldT, FieldType))
8688 if (F.isBitField()) {
8689 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8692 if (!this->emitInitField(*FieldT, F.Offset, E))
8696 if (!this->emitGetPtrField(F.Offset, E))
8698 if (!emitHLSLConstructAggregate(FieldType, Elements, ElemIdx, E))
8700 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()
bool isIndeterminate() const
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.
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
const APValue * evaluateValue() const
Attempt to evaluate the value of the initializer attached to this declaration, and produce notes expl...
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 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 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 visitAPValue(const APValue &Val, PrimType ValType, SourceInfo Info)
Visit an APValue.
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.
bool visitAPValueInitializer(const APValue &Val, SourceInfo Info, QualType T)
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.
const Expr * asExpr() const
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
EvalResult is a struct with detailed info about an evaluated expression.
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()