19#include "llvm/Support/SaveAndRestore.h"
30 if (
const auto *CE = dyn_cast_if_present<ConstantExpr>(E);
31 CE && CE->hasAPValueResult() &&
33 return CE->getResultAsAPSInt().getBoolValue();
44 OldInitializingDecl(
Ctx->InitializingDecl) {
45 Ctx->InitializingDecl = VD;
50 this->
Ctx->InitializingDecl = OldInitializingDecl;
51 this->
Ctx->InitStack.pop_back();
64 bool NewInitializing,
bool NewToLValue)
65 : Ctx(Ctx), OldDiscardResult(Ctx->DiscardResult),
66 OldInitializing(Ctx->
Initializing), OldToLValue(Ctx->ToLValue) {
67 Ctx->DiscardResult = NewDiscardResult;
68 Ctx->Initializing = NewInitializing;
69 Ctx->ToLValue = NewToLValue;
73 Ctx->DiscardResult = OldDiscardResult;
74 Ctx->Initializing = OldInitializing;
75 Ctx->ToLValue = OldToLValue;
82 bool OldDiscardResult;
87template <
class Emitter>
91 return Ctx->emitThis(E);
94 return Ctx->emitGetPtrFieldPop(
Offset, E);
96 return Ctx->emitGetPtrLocal(
Offset, E);
100 if (!Ctx->emitConstUint32(
Offset, E))
102 return Ctx->emitArrayElemPtrPopUint32(E);
104 return Ctx->emitRVOPtr(E);
108 llvm_unreachable(
"Unhandled InitLink kind");
124 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
125 assert(LI.Name != Name);
128 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel, ContinueLabel,
149 : Ctx(Ctx), OldCaseLabels(
std::move(this->Ctx->CaseLabels)) {
151 for (
const LabelInfo &LI : Ctx->LabelInfoStack)
152 assert(LI.Name != Name);
155 this->Ctx->CaseLabels = std::move(CaseLabels);
156 this->Ctx->LabelInfoStack.emplace_back(Name, BreakLabel,
158 DefaultLabel, Ctx->VarScope);
162 this->Ctx->CaseLabels = std::move(OldCaseLabels);
163 this->Ctx->LabelInfoStack.pop_back();
168 CaseMap OldCaseLabels;
174 Ctx->InStmtExpr =
true;
192 : Ctx(Ctx), OldFlag(Ctx->LocOverride), Enabled(Enabled) {
195 Ctx->LocOverride = NewValue;
200 Ctx->LocOverride = OldFlag;
205 std::optional<SourceInfo> OldFlag;
212template <
class Emitter>
220 case CK_LValueToRValue: {
231 if (
const auto *DRE = dyn_cast<DeclRefExpr>(SubExpr)) {
237 if (
auto GlobalIndex =
P.getGlobal(D))
238 return this->emitGetGlobal(*SubExprT, *GlobalIndex, E);
239 }
else if (
auto It =
Locals.find(D); It !=
Locals.end()) {
240 return this->emitGetLocal(*SubExprT, It->second.Offset, E);
241 }
else if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
242 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
243 return this->emitGetParam(*SubExprT, It->second.Index, E);
255 if (!this->emitGetPtrLocal(*LocalIndex, E))
259 if (!this->
visit(SubExpr))
263 return this->emitLoadPop(*SubExprT, E);
268 return this->emitMemcpy(E);
271 case CK_DerivedToBaseMemberPointer: {
282 ->getMostRecentCXXRecordDecl();
284 const CXXRecordDecl *ToDecl = B->getType()->getAsCXXRecordDecl();
285 unsigned DerivedOffset =
Ctx.collectBaseOffset(ToDecl, CurDecl);
287 if (!this->emitCastMemberPtrBasePop(DerivedOffset, ToDecl, E))
295 case CK_BaseToDerivedMemberPointer: {
306 ->getMostRecentCXXRecordDecl();
310 typedef std::reverse_iterator<CastExpr::path_const_iterator> ReverseIter;
312 PathI != PathE; ++PathI) {
313 const CXXRecordDecl *ToDecl = (*PathI)->getType()->getAsCXXRecordDecl();
314 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
316 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
323 assert(ToDecl != CurDecl);
324 unsigned DerivedOffset =
Ctx.collectBaseOffset(CurDecl, ToDecl);
326 if (!this->emitCastMemberPtrDerivedPop(-DerivedOffset, ToDecl, E))
332 case CK_UncheckedDerivedToBase:
333 case CK_DerivedToBase: {
338 if (
const auto *PT = dyn_cast<PointerType>(Ty))
339 return PT->getPointeeType()->getAsCXXRecordDecl();
340 return Ty->getAsCXXRecordDecl();
347 if (B->isVirtual()) {
348 if (!this->emitGetPtrVirtBasePop(extractRecordDecl(B->getType()), E))
350 CurType = B->getType();
352 unsigned DerivedOffset = collectBaseOffset(B->getType(), CurType);
353 if (!this->emitGetPtrBasePop(
356 CurType = B->getType();
363 case CK_BaseToDerived: {
366 unsigned DerivedOffset =
372 return this->emitGetPtrDerivedPop(DerivedOffset,
377 case CK_FloatingCast: {
380 return this->emitVectorConversion(E->
getSubExpr(), E);
384 if (!this->
visit(SubExpr))
386 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
390 case CK_IntegralToFloating: {
392 return this->emitVectorConversion(E->
getSubExpr(), E);
395 if (!this->
visit(SubExpr))
397 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
398 return this->emitCastIntegralFloating(
classifyPrim(SubExpr),
399 TargetSemantics, getFPOptions(E), E);
402 case CK_FloatingToBoolean: {
404 return this->emitVectorConversion(E->
getSubExpr(), E);
408 if (
const auto *FL = dyn_cast<FloatingLiteral>(SubExpr))
409 return this->emitConstBool(FL->getValue().isNonZero(), E);
410 if (!this->
visit(SubExpr))
412 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
415 case CK_FloatingToIntegral: {
417 return this->emitVectorConversion(E->
getSubExpr(), E);
420 if (!this->
visit(SubExpr))
424 return this->emitCastFloatingIntegralAP(
Ctx.getBitWidth(E->
getType()),
427 return this->emitCastFloatingIntegralAPS(
Ctx.getBitWidth(E->
getType()),
430 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
433 case CK_NullToPointer:
434 case CK_NullToMemberPointer: {
439 if (!PointeeType.
isNull()) {
441 Desc =
P.createDescriptor(SubExpr, *T);
443 Desc =
P.createDescriptor(SubExpr, PointeeType.
getTypePtr(),
447 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
451 case CK_PointerToIntegral: {
452 if (!this->
visit(SubExpr))
458 if (!this->emitDecayPtr(FromT,
PT_Ptr, E))
464 return this->emitCastPointerIntegralAP(
Ctx.getBitWidth(E->
getType()), E);
466 return this->emitCastPointerIntegralAPS(
Ctx.getBitWidth(E->
getType()), E);
467 return this->emitCastPointerIntegral(T, E);
470 case CK_ArrayToPointerDecay: {
471 if (!this->
visit(SubExpr))
473 return this->emitArrayDecay(E);
476 case CK_IntegralToPointer: {
478 assert(IntType->isIntegralOrEnumerationType());
479 if (!this->
visit(SubExpr))
487 Desc =
P.createDescriptor(SubExpr, *T);
494 if (!this->emitGetIntPtr(T, Desc, E))
502 return this->emitDecayPtr(
PT_Ptr, DestPtrT, E);
505 case CK_AtomicToNonAtomic:
506 case CK_ConstructorConversion:
507 case CK_FunctionToPointerDecay:
508 case CK_NonAtomicToAtomic:
510 case CK_UserDefinedConversion:
511 case CK_AddressSpaceConversion:
512 case CK_CPointerToObjCPointerCast:
529 return this->emitBuiltinBitCast(E);
544 if (!this->
visit(SubExpr))
552 return this->emitFnPtrCast(E);
559 if (!this->
visit(SubExpr))
561 return this->emitDecayPtr(*FromT, *ToT, E);
563 case CK_IntegralToBoolean:
564 case CK_FixedPointToBoolean: {
566 return this->emitVectorConversion(E->
getSubExpr(), E);
572 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr))
573 return this->emitConst(IL->getValue(), E);
574 if (!this->
visit(SubExpr))
579 case CK_IntegralCast:
581 return this->emitVectorConversion(E->
getSubExpr(), E);
583 case CK_BooleanToSignedIntegral: {
590 if (
const auto *IL = dyn_cast<IntegerLiteral>(SubExpr)) {
595 if (!this->emitConst(IL->getValue(), SubExpr))
598 if (!this->
visit(SubExpr))
606 if (!ED->isFixed()) {
607 if (!this->emitCheckEnumValue(*FromT, ED, E))
613 if (!this->emitCastAP(*FromT,
Ctx.getBitWidth(E->
getType()), E))
616 if (!this->emitCastAPS(*FromT,
Ctx.getBitWidth(E->
getType()), E))
621 if (!this->emitCast(*FromT, *ToT, E))
624 if (E->
getCastKind() == CK_BooleanToSignedIntegral)
625 return this->emitNeg(*ToT, E);
629 case CK_PointerToBoolean:
630 case CK_MemberPointerToBoolean: {
633 if (!this->
visit(SubExpr))
635 return this->emitIsNonNull(PtrT, E);
638 case CK_IntegralComplexToBoolean:
639 case CK_FloatingComplexToBoolean: {
640 if (!this->
visit(SubExpr))
642 return this->emitComplexBoolCast(SubExpr);
645 case CK_IntegralComplexToReal:
646 case CK_FloatingComplexToReal:
647 return this->emitComplexReal(SubExpr);
649 case CK_IntegralRealToComplex:
650 case CK_FloatingRealToComplex: {
657 if (!this->emitGetPtrLocal(*LocalIndex, E))
666 if (!this->visitZeroInitializer(T, SubExpr->
getType(), SubExpr))
668 return this->emitInitElem(T, 1, SubExpr);
671 case CK_IntegralComplexCast:
672 case CK_FloatingComplexCast:
673 case CK_IntegralComplexToFloatingComplex:
674 case CK_FloatingComplexToIntegralComplex: {
681 if (!this->emitGetPtrLocal(*LocalIndex, E))
688 unsigned SubExprOffset =
690 if (!this->
visit(SubExpr))
692 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
700 for (
unsigned I = 0; I != 2; ++I) {
701 if (!this->emitGetLocal(
PT_Ptr, SubExprOffset, E))
703 if (!this->emitArrayElemPop(SourceElemT, I, E))
707 if (!this->emitPrimCast(SourceElemT, DestElemT, DestElemType, E))
711 if (!this->emitInitElem(DestElemT, I, E))
717 case CK_VectorSplat: {
728 if (!this->emitGetPtrLocal(*LocalIndex, E))
734 unsigned ElemOffset =
738 if (!this->
visit(SubExpr))
743 if (!this->emitSetLocal(ElemT, ElemOffset, E))
746 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
747 if (!this->emitGetLocal(ElemT, ElemOffset, E))
749 if (!this->emitInitElem(ElemT, I, E))
756 case CK_HLSLVectorTruncation: {
761 if (!this->
visit(SubExpr))
763 return this->emitArrayElemPop(*ResultT, 0, E);
772 if (!this->emitGetPtrLocal(*LocalIndex, E))
777 if (!this->
visit(SubExpr))
779 return this->emitCopyArray(classifyVectorElementType(E->
getType()), 0, 0,
783 case CK_IntegralToFixedPoint: {
784 if (!this->
visit(SubExpr))
788 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
793 return this->emitPopFixedPoint(E);
796 case CK_FloatingToFixedPoint: {
797 if (!this->
visit(SubExpr))
801 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
802 if (!this->emitCastFloatingFixedPoint(Sem, E))
805 return this->emitPopFixedPoint(E);
808 case CK_FixedPointToFloating: {
809 if (!this->
visit(SubExpr))
811 const auto *TargetSemantics = &
Ctx.getFloatSemantics(E->
getType());
812 if (!this->emitCastFixedPointFloating(TargetSemantics, E))
815 return this->emitPopFloat(E);
818 case CK_FixedPointToIntegral: {
819 if (!this->
visit(SubExpr))
822 if (!this->emitCastFixedPointIntegral(IntegralT, E))
825 return this->emitPop(IntegralT, E);
828 case CK_FixedPointCast: {
829 if (!this->
visit(SubExpr))
832 Ctx.getASTContext().getFixedPointSemantics(E->
getType()).toOpaqueInt();
833 if (!this->emitCastFixedPoint(Sem, E))
836 return this->emitPopFixedPoint(E);
848 case CK_LValueBitCast:
853 case CK_HLSLArrayRValue: {
860 if (!this->emitGetPtrLocal(*LocalIndex, E))
863 if (!this->
visit(SubExpr))
865 return this->emitMemcpy(E);
868 case CK_HLSLMatrixTruncation: {
873 if (!this->
visit(SubExpr))
875 return this->emitArrayElemPop(*ResultT, 0, E);
884 if (!this->emitGetPtrLocal(*LocalIndex, E))
889 if (!this->
visit(SubExpr))
891 return this->emitCopyArray(classifyMatrixElementType(SubExpr->
getType()), 0,
895 case CK_HLSLAggregateSplatCast: {
905 if (!this->emitGetPtrLocal(*LocalIndex, E))
915 if (!this->
visit(SubExpr))
917 if (!this->emitSetLocal(SrcElemT, SrcOffset, E))
921 return emitHLSLAggregateSplat(SrcElemT, SrcOffset, E->
getType(), E);
924 case CK_HLSLElementwiseCast: {
935 unsigned SrcPtrOffset =
937 if (!this->
visit(SubExpr))
939 if (!this->emitSetLocal(
PT_Ptr, SrcPtrOffset, E))
943 if (!emitHLSLFlattenAggregate(SrcType, SrcPtrOffset, Elements, 1, E))
945 if (Elements.empty())
948 const HLSLFlatElement &Src = Elements[0];
949 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
951 return this->emitPrimCast(Src.Type, *DestT, DestType, E);
958 if (!this->emitGetPtrLocal(*LocalIndex, E))
964 if (!this->
visit(SubExpr))
966 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
970 unsigned ElemCount = countHLSLFlatElements(DestType);
973 Elements.reserve(ElemCount);
974 if (!emitHLSLFlattenAggregate(SrcType, SrcOffset, Elements, ElemCount, E))
979 assert(Elements.size() == ElemCount &&
980 "Source type has fewer scalar elements than the destination type");
982 return emitHLSLConstructAggregate(DestType, Elements, E);
989 const Record::Field *RF = R->getField(UnionField);
990 QualType FieldType = RF->Decl->getType();
993 if (!this->
visit(SubExpr))
995 if (RF->isBitField())
996 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
998 return this->emitInitFieldActivate(*PT, RF->Offset, E);
1001 if (!this->emitGetPtrField(RF->Offset, E))
1003 if (!this->emitActivate(E))
1009 return this->emitInvalid(E);
1011 llvm_unreachable(
"Unhandled clang::CastKind enum");
1014template <
class Emitter>
1016 return this->emitBuiltinBitCast(E);
1019template <
class Emitter>
1024 return this->emitConst(
LE->getValue(),
LE);
1027template <
class Emitter>
1033 return this->emitFloat(F, E);
1036template <
class Emitter>
1046 if (!this->emitGetPtrLocal(*LocalIndex, E))
1053 if (!this->visitZeroInitializer(SubExprT, SubExpr->
getType(), SubExpr))
1055 if (!this->emitInitElem(SubExprT, 0, SubExpr))
1060template <
class Emitter>
1068 auto Sem =
Ctx.getASTContext().getFixedPointSemantics(E->
getType());
1073template <
class Emitter>
1078template <
class Emitter>
1105 return this->emitComplexComparison(LHS, RHS, E);
1113 if (!this->
visit(LHS))
1116 if (!this->
visit(RHS))
1119 if (!this->emitToMemberPtr(E))
1125 if (!this->emitCastMemberPtrPtr(E))
1142 Ctx.getASTContext().CompCategories.lookupInfoForType(E->
getType());
1148 if (!this->emitGetPtrLocal(*ResultIndex, E))
1155 return this->emitCMP3(*
LT, CmpInfo, E);
1158 if (!
LT || !RT || !T)
1168 return this->visitAssignment(LHS, RHS, E);
1175 auto MaybeCastToBool = [
this, T, E](
bool Result) {
1179 return this->emitPopBool(E);
1181 return this->emitCast(
PT_Bool, *T, E);
1185 auto Discard = [
this, T, E](
bool Result) {
1193 return MaybeCastToBool(this->emitEQ(*
LT, E));
1195 return MaybeCastToBool(this->emitNE(*
LT, E));
1197 return MaybeCastToBool(this->emitLT(*
LT, E));
1199 return MaybeCastToBool(this->emitLE(*
LT, E));
1201 return MaybeCastToBool(this->emitGT(*
LT, E));
1203 return MaybeCastToBool(this->emitGE(*
LT, E));
1206 return Discard(this->emitSubf(getFPOptions(E), E));
1207 return Discard(this->emitSub(*T, E));
1210 return Discard(this->emitAddf(getFPOptions(E), E));
1211 return Discard(this->emitAdd(*T, E));
1214 return Discard(this->emitMulf(getFPOptions(E), E));
1215 return Discard(this->emitMul(*T, E));
1217 return Discard(this->emitRem(*T, E));
1220 return Discard(this->emitDivf(getFPOptions(E), E));
1221 return Discard(this->emitDiv(*T, E));
1223 return Discard(this->emitBitAnd(*T, E));
1225 return Discard(this->emitBitOr(*T, E));
1227 return Discard(this->emitShl(*
LT, *RT, E));
1229 return Discard(this->emitShr(*
LT, *RT, E));
1231 return Discard(this->emitBitXor(*T, E));
1234 llvm_unreachable(
"Already handled earlier");
1239 llvm_unreachable(
"Unhandled binary op");
1244template <
class Emitter>
1250 if ((Op != BO_Add && Op != BO_Sub) ||
1261 auto visitAsPointer = [&](
const Expr *E,
PrimType T) ->
bool {
1262 if (!this->
visit(E))
1265 return this->emitDecayPtr(T,
PT_Ptr, E);
1274 if (!visitAsPointer(RHS, *RT) || !visitAsPointer(LHS, *
LT))
1282 ElemTypeSize =
Ctx.getASTContext().getTypeSizeInChars(ElemType);
1285 if (!this->emitSubPtr(IntT, ElemTypeSize.
isZero(), E))
1292 if (!visitAsPointer(RHS, *RT))
1294 if (!this->
visit(LHS))
1298 if (!visitAsPointer(LHS, *
LT))
1300 if (!this->
visit(RHS))
1311 if (!this->emitAddOffset(OffsetType, E))
1315 if (!this->emitSubOffset(OffsetType, E))
1332template <
class Emitter>
1342 LabelTy LabelTrue = this->getLabel();
1343 LabelTy LabelEnd = this->getLabel();
1347 if (!this->jumpTrue(LabelTrue, E))
1352 if (!this->jump(LabelEnd, E))
1355 this->emitLabel(LabelTrue);
1356 this->emitConstBool(
true, E);
1357 this->fallthrough(LabelEnd);
1358 this->emitLabel(LabelEnd);
1361 assert(Op == BO_LAnd);
1364 LabelTy LabelFalse = this->getLabel();
1365 LabelTy LabelEnd = this->getLabel();
1369 if (!this->jumpFalse(LabelFalse, E))
1374 if (!this->jump(LabelEnd, E))
1377 this->emitLabel(LabelFalse);
1378 this->emitConstBool(
false, E);
1379 this->fallthrough(LabelEnd);
1380 this->emitLabel(LabelEnd);
1384 return this->emitPopBool(E);
1389 return this->emitCast(
PT_Bool, *T, E);
1393template <
class Emitter>
1400 if (!this->emitGetPtrLocal(*LocalIndex, E))
1409 PrimType ResultElemT = this->classifyComplexElementType(E->
getType());
1410 unsigned ResultOffset = ~0u;
1416 if (!this->emitDupPtr(E))
1418 if (!this->emitSetLocal(
PT_Ptr, ResultOffset, E))
1423 LHSType = AT->getValueType();
1426 RHSType = AT->getValueType();
1435 if (Op == BO_Mul && LHSIsComplex && RHSIsComplex) {
1440 if (!this->
visit(LHS))
1442 if (!this->
visit(RHS))
1444 if (!this->emitMulc(ElemT, E))
1447 return this->emitPopPtr(E);
1451 if (Op == BO_Div && RHSIsComplex) {
1458 if (!LHSIsComplex) {
1463 LHSOffset = *LocalIndex;
1465 if (!this->emitGetPtrLocal(LHSOffset, E))
1468 if (!this->
visit(LHS))
1471 if (!this->emitInitElem(ElemT, 0, E))
1474 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1476 if (!this->emitInitElem(ElemT, 1, E))
1479 if (!this->
visit(LHS))
1483 if (!this->
visit(RHS))
1485 if (!this->emitDivc(ElemT, E))
1488 return this->emitPopPtr(E);
1495 if (!this->
visit(LHS))
1497 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1502 if (!this->
visit(LHS))
1504 if (!this->emitSetLocal(LHST, LHSOffset, E))
1512 if (!this->
visit(RHS))
1514 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1519 if (!this->
visit(RHS))
1521 if (!this->emitSetLocal(RHST, RHSOffset, E))
1528 auto loadComplexValue = [
this](
bool IsComplex,
bool LoadZero,
1529 unsigned ElemIndex,
unsigned Offset,
1530 const Expr *E) ->
bool {
1532 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1534 return this->emitArrayElemPop(classifyComplexElementType(E->
getType()),
1537 if (ElemIndex == 0 || !LoadZero)
1544 for (
unsigned ElemIndex = 0; ElemIndex != 2; ++ElemIndex) {
1547 if (!this->emitGetLocal(
PT_Ptr, ResultOffset, E))
1554 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1557 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1560 if (!this->emitAddf(getFPOptions(E), E))
1563 if (!this->emitAdd(ResultElemT, E))
1568 if (!loadComplexValue(LHSIsComplex,
true, ElemIndex, LHSOffset, LHS))
1571 if (!loadComplexValue(RHSIsComplex,
true, ElemIndex, RHSOffset, RHS))
1574 if (!this->emitSubf(getFPOptions(E), E))
1577 if (!this->emitSub(ResultElemT, E))
1582 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1585 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1589 if (!this->emitMulf(getFPOptions(E), E))
1592 if (!this->emitMul(ResultElemT, E))
1597 assert(!RHSIsComplex);
1598 if (!loadComplexValue(LHSIsComplex,
false, ElemIndex, LHSOffset, LHS))
1601 if (!loadComplexValue(RHSIsComplex,
false, ElemIndex, RHSOffset, RHS))
1605 if (!this->emitDivf(getFPOptions(E), E))
1608 if (!this->emitDiv(ResultElemT, E))
1619 if (!this->emitInitElemPop(ResultElemT, ElemIndex, E))
1622 if (!this->emitPop(ResultElemT, E))
1627 return this->emitPopPtr(E);
1633template <
class Emitter>
1638 "Comma op should be handled in VisitBinaryOperator");
1652 if (!this->emitGetPtrLocal(*LocalIndex, E))
1666 assert(
Ctx.getASTContext().hasSameUnqualifiedType(
1669 if (!this->
visit(LHS))
1671 if (!this->
visit(RHS))
1673 if (!this->emitCopyArray(ElemT, 0, 0, VecTy->getNumElements(), E))
1676 return this->emitPopPtr(E);
1681 unsigned LHSOffset =
1683 if (!this->
visit(LHS))
1685 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
1689 unsigned RHSOffset =
1691 if (!this->
visit(RHS))
1693 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
1705 if (NeedIntPromot) {
1707 Ctx.getASTContext().getPromotedIntegerType(
Ctx.getASTContext().BoolTy);
1712 auto getElem = [=](
unsigned Offset,
PrimType ElemT,
unsigned Index) {
1713 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
1715 if (!this->emitArrayElemPop(ElemT, Index, E))
1718 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
1720 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1722 }
else if (NeedIntPromot) {
1723 if (!this->emitPrimCast(ElemT, PromotT, PromotTy, E))
1729#define EMIT_ARITH_OP(OP) \
1731 if (ElemT == PT_Float) { \
1732 if (!this->emit##OP##f(getFPOptions(E), E)) \
1735 if (!this->emit##OP(ElemT, E)) \
1741 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
1742 if (!getElem(LHSOffset, ElemT, I))
1744 if (!getElem(RHSOffset, RHSElemT, I))
1756 if (!this->emitRem(ElemT, E))
1760 if (!this->emitBitAnd(OpT, E))
1764 if (!this->emitBitOr(OpT, E))
1768 if (!this->emitBitXor(OpT, E))
1772 if (!this->emitShl(OpT, RHSElemT, E))
1776 if (!this->emitShr(OpT, RHSElemT, E))
1780 if (!this->emitEQ(ElemT, E))
1784 if (!this->emitNE(ElemT, E))
1788 if (!this->emitLE(ElemT, E))
1792 if (!this->emitLT(ElemT, E))
1796 if (!this->emitGE(ElemT, E))
1800 if (!this->emitGT(ElemT, E))
1805 if (!this->emitBitAnd(ResultElemT, E))
1810 if (!this->emitBitOr(ResultElemT, E))
1814 return this->emitInvalid(E);
1823 if (!this->emitPrimCast(
PT_Bool, ResultElemT, VecTy->getElementType(), E))
1825 if (!this->emitNeg(ResultElemT, E))
1831 if (NeedIntPromot &&
1832 !this->emitPrimCast(PromotT, ResultElemT, VecTy->getElementType(), E))
1836 if (!this->emitInitElem(ResultElemT, I, E))
1845template <
class Emitter>
1855 auto LHSSemaInt = LHSSema.toOpaqueInt();
1857 auto RHSSemaInt = RHSSema.toOpaqueInt();
1859 if (!this->
visit(LHS))
1867 if (!this->
visit(RHS))
1876 auto ConvertResult = [&](
bool R) ->
bool {
1880 auto CommonSema = LHSSema.getCommonSemantics(RHSSema).toOpaqueInt();
1881 if (ResultSema != CommonSema)
1882 return this->emitCastFixedPoint(ResultSema, E);
1886 auto MaybeCastToBool = [&](
bool Result) {
1891 return this->emitPop(T, E);
1893 return this->emitCast(
PT_Bool, T, E);
1899 return MaybeCastToBool(this->emitEQFixedPoint(E));
1901 return MaybeCastToBool(this->emitNEFixedPoint(E));
1903 return MaybeCastToBool(this->emitLTFixedPoint(E));
1905 return MaybeCastToBool(this->emitLEFixedPoint(E));
1907 return MaybeCastToBool(this->emitGTFixedPoint(E));
1909 return MaybeCastToBool(this->emitGEFixedPoint(E));
1911 return ConvertResult(this->emitAddFixedPoint(E));
1913 return ConvertResult(this->emitSubFixedPoint(E));
1915 return ConvertResult(this->emitMulFixedPoint(E));
1917 return ConvertResult(this->emitDivFixedPoint(E));
1919 return ConvertResult(this->emitShiftFixedPoint(
true, E));
1921 return ConvertResult(this->emitShiftFixedPoint(
false, E));
1924 return this->emitInvalid(E);
1927 llvm_unreachable(
"unhandled binop opcode");
1930template <
class Emitter>
1939 if (!this->
visit(SubExpr))
1941 if (!this->emitNegFixedPoint(E))
1944 return this->emitPopFixedPoint(E);
1950 llvm_unreachable(
"Unhandled unary opcode");
1953template <
class Emitter>
1962 return this->visitZeroInitializer(*T, QT, E);
1970 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1971 CXXRD && CXXRD->getNumVBases() > 0) {
1981 return this->visitZeroRecordInitializer(R, E);
1988 return this->visitZeroArrayInitializer(QT, E);
1992 QualType ElemQT = ComplexTy->getElementType();
1994 for (
unsigned I = 0; I < 2; ++I) {
1995 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
1997 if (!this->emitInitElem(ElemT, I, E))
2004 unsigned NumVecElements = VecT->getNumElements();
2005 QualType ElemQT = VecT->getElementType();
2008 for (
unsigned I = 0; I < NumVecElements; ++I) {
2009 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2011 if (!this->emitInitElem(ElemT, I, E))
2018 unsigned NumElems = MT->getNumElementsFlattened();
2019 QualType ElemQT = MT->getElementType();
2022 for (
unsigned I = 0; I != NumElems; ++I) {
2023 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2025 if (!this->emitInitElem(ElemT, I, E))
2034template <
class Emitter>
2047 for (
const Expr *SubExpr : {LHS, RHS}) {
2048 if (!this->
visit(SubExpr)) {
2055 if (SubExpr ==
Base &&
Base->getType()->isPointerType()) {
2056 if (!this->emitExpandPtr(E))
2067 return this->emitError(E);
2070 if (!this->emitFlip(
PT_Ptr, *IndexT, E))
2074 if (!this->emitArrayElemPtrPop(*IndexT, E))
2077 return this->emitPopPtr(E);
2083 return this->emitLoadPop(*T, E);
2086template <
class Emitter>
2088 const Expr *ArrayFiller,
const Expr *E) {
2093 QT = AT->getValueType();
2096 if (
Inits.size() == 0)
2098 return this->emitInvalid(E);
2113 if (
Inits.size() == 0)
2114 return this->visitZeroInitializer(*T, QT, E);
2115 assert(
Inits.size() == 1);
2128 auto initPrimitiveField = [=](
const Record::Field *FieldToInit,
2135 bool BitField = FieldToInit->isBitField();
2137 return this->emitInitBitFieldActivate(T, FieldToInit->Offset,
2138 FieldToInit->bitWidth(), E);
2140 return this->emitInitBitField(T, FieldToInit->Offset,
2141 FieldToInit->bitWidth(), E);
2143 return this->emitInitFieldActivate(T, FieldToInit->Offset, E);
2144 return this->emitInitField(T, FieldToInit->Offset, E);
2147 auto initCompositeField = [=](
const Record::Field *FieldToInit,
2155 if (!this->emitGetPtrField(FieldToInit->Offset,
Init))
2158 if (
Activate && !this->emitActivate(E))
2165 if (
Inits.size() == 0) {
2166 if (!this->visitZeroRecordInitializer(R, E))
2171 if (
const auto *ILE = dyn_cast<InitListExpr>(E))
2172 FToInit = ILE->getInitializedFieldInUnion();
2176 const Record::Field *FieldToInit = R->getField(FToInit);
2178 if (!initPrimitiveField(FieldToInit,
Init, *T,
true))
2181 if (!initCompositeField(FieldToInit,
Init,
true))
2185 return this->emitFinishInit(E);
2188 assert(!R->isUnion());
2189 unsigned InitIndex = 0;
2192 while (InitIndex < R->getNumFields() &&
2193 R->getField(InitIndex)->isUnnamedBitField())
2204 const Record::Field *FieldToInit = R->getField(InitIndex);
2205 if (!initPrimitiveField(FieldToInit,
Init, *T))
2210 if (
const Record::Base *B = R->getBase(
Init->getType())) {
2211 if (!this->emitGetPtrBase(B->Offset,
Init))
2219 const Record::Field *FieldToInit = R->getField(InitIndex);
2220 if (!initCompositeField(FieldToInit,
Init))
2226 return this->emitFinishInit(E);
2231 Ctx.getASTContext().getAsConstantArrayType(QT);
2234 if (
Initializing && !this->emitCheckArrayDestSize(NumElems, E))
2237 if (
Inits.size() == 1 && QT ==
Inits[0]->getType())
2241 unsigned ElementIndex = 0;
2243 if (
const auto *EmbedS =
2244 dyn_cast<EmbedExpr>(
Init->IgnoreParenImpCasts())) {
2252 if (!this->emitCastIntegralFloating(
classifyPrim(IL), Sem,
2253 getFPOptions(E), E))
2259 return this->emitInitElem(TargetT, ElemIndex, IL);
2261 if (!EmbedS->doForEachDataElement(Eval, ElementIndex))
2277 for (; ElementIndex != NumElems; ++ElementIndex) {
2283 return this->emitFinishInit(E);
2287 unsigned NumInits =
Inits.size();
2292 QualType ElemQT = ComplexTy->getElementType();
2294 if (NumInits == 0) {
2296 for (
unsigned I = 0; I < 2; ++I) {
2297 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2299 if (!this->emitInitElem(ElemT, I, E))
2302 }
else if (NumInits == 2) {
2303 unsigned InitIndex = 0;
2308 if (!this->emitInitElem(ElemT, InitIndex, E))
2317 unsigned NumVecElements = VecT->getNumElements();
2318 assert(NumVecElements >=
Inits.size());
2320 QualType ElemQT = VecT->getElementType();
2324 unsigned InitIndex = 0;
2331 if (
const auto *InitVecT =
Init->getType()->getAs<
VectorType>()) {
2332 if (!this->emitCopyArray(ElemT, 0, InitIndex,
2333 InitVecT->getNumElements(), E))
2335 InitIndex += InitVecT->getNumElements();
2337 if (!this->emitInitElem(ElemT, InitIndex, E))
2343 assert(InitIndex <= NumVecElements);
2346 for (; InitIndex != NumVecElements; ++InitIndex) {
2347 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
2349 if (!this->emitInitElem(ElemT, InitIndex, E))
2356 unsigned NumElems = MT->getNumElementsFlattened();
2357 assert(
Inits.size() == NumElems);
2359 QualType ElemQT = MT->getElementType();
2365 for (
unsigned I = 0; I != NumElems; ++I) {
2368 if (!this->emitInitElem(ElemT, I, E))
2379template <
class Emitter>
2386 return this->emitInitElem(*InitT, ElemIndex,
Init);
2392 if (!this->emitConstUint32(ElemIndex,
Init))
2394 if (!this->emitArrayElemPtrUint32(
Init))
2399template <
class Emitter>
2402 bool Activate,
bool IsOperatorCall) {
2404 llvm::BitVector NonNullArgs;
2405 if (FuncDecl && FuncDecl->
hasAttr<NonNullAttr>())
2408 bool ExplicitMemberFn =
false;
2409 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl))
2410 ExplicitMemberFn = MD->isExplicitObjectMemberFunction();
2412 unsigned ArgIndex = 0;
2413 for (
const Expr *Arg : Args) {
2415 if (!this->
visit(Arg))
2423 unsigned DeclIndex = ArgIndex - IsOperatorCall + ExplicitMemberFn;
2424 if (DeclIndex < FuncDecl->getNumParams())
2425 Source = FuncDecl->
getParamDecl(ArgIndex - IsOperatorCall +
2434 if (!this->emitGetPtrLocal(*LocalIndex, Arg))
2442 if (!this->emitActivate(Arg))
2446 if (!NonNullArgs.empty() && NonNullArgs[ArgIndex]) {
2449 if (!this->emitCheckNonNullArg(ArgT, Arg))
2460template <
class Emitter>
2465template <
class Emitter>
2471template <
class Emitter>
2477template <
class Emitter>
2495template <
class Emitter>
2497 auto It = E->
begin();
2498 return this->
visit(*It);
2503 bool AlignOfReturnsPreferred =
2510 T = Ref->getPointeeType();
2512 if (T.getQualifiers().hasUnaligned())
2518 if (Kind == UETT_PreferredAlignOf || AlignOfReturnsPreferred)
2524template <
class Emitter>
2531 if (Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
2537 ArgType = Ref->getPointeeType();
2543 if (
ArgType->isDependentType() || !
ArgType->isConstantSizeType())
2544 return this->emitInvalid(E);
2546 if (Kind == UETT_SizeOf)
2555 return this->emitConst(Size.getQuantity(), E);
2558 if (Kind == UETT_CountOf) {
2564 if (
const auto *CAT =
2568 return this->emitConst(CAT->getSize(), E);
2578 if (VAT->getElementType()->isArrayType()) {
2579 std::optional<APSInt> Res =
2581 ? VAT->getSizeExpr()->getIntegerConstantExpr(ASTCtx)
2586 return this->emitConst(*Res, E);
2591 if (Kind == UETT_AlignOf || Kind == UETT_PreferredAlignOf) {
2611 if (
const auto *DRE = dyn_cast<DeclRefExpr>(Arg))
2614 else if (
const auto *ME = dyn_cast<MemberExpr>(Arg))
2624 return this->emitConst(Size.getQuantity(), E);
2627 if (Kind == UETT_VectorElements) {
2632 return this->emitConst(VT->getNumElements(), E);
2634 return this->emitSizelessVectorElementSize(E);
2637 if (Kind == UETT_VecStep) {
2639 unsigned N = VT->getNumElements();
2646 return this->emitConst(N, E);
2648 return this->emitConst(1, E);
2651 if (Kind == UETT_OpenMPRequiredSimdAlign) {
2660 if (Kind == UETT_PtrAuthTypeDiscriminator) {
2662 return this->emitInvalid(E);
2664 return this->emitConst(
2665 const_cast<ASTContext &
>(ASTCtx).getPointerAuthTypeDiscriminator(
2673template <
class Emitter>
2682 if (
const auto *VD = dyn_cast<VarDecl>(
Member)) {
2685 if (
auto GlobalIndex =
P.getGlobal(VD)) {
2686 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
2688 if (
Member->getType()->isReferenceType())
2689 return this->emitLoadPopPtr(E);
2698 if (
const auto *MD = dyn_cast<CXXMethodDecl>(
Member);
2699 MD && !MD->isStatic()) {
2703 if (!this->
discard(Base) && !this->emitSideEffect(E))
2718 const Record::Field *F = R->getField(FD);
2722 const auto maybeLoadValue = [&]() ->
bool {
2726 return this->emitLoadPop(*T, E);
2731 if (F->Decl->getType()->isReferenceType())
2732 return this->emitGetFieldPop(
PT_Ptr, F->Offset, E) && maybeLoadValue();
2733 return this->emitGetPtrFieldPop(F->Offset, E) && maybeLoadValue();
2736template <
class Emitter>
2746template <
class Emitter>
2760 if (!this->
visit(Common))
2762 return this->emitCopyArray(*SubExprT, 0, 0, Size, E);
2776 for (
size_t I = 0; I != Size; ++I) {
2788template <
class Emitter>
2803 return this->emitGetLocal(SubExprT, It->second, E);
2806 if (!this->
visit(SourceExpr))
2813 if (!this->emitSetLocal(SubExprT, LocalIndex, E))
2822 return this->emitGetLocal(SubExprT, LocalIndex, E);
2826template <
class Emitter>
2839 bool IsBcpCall =
false;
2840 if (
const auto *CE = dyn_cast<CallExpr>(
Condition->IgnoreParenCasts());
2841 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_constant_p) {
2845 LabelTy LabelEnd = this->getLabel();
2846 LabelTy LabelFalse = this->getLabel();
2849 if (!this->emitPushIgnoreDiags(E))
2857 if (this->checkingForUndefinedBehavior()) {
2860 if (!this->
discard(FalseExpr))
2877 if (!this->jumpFalse(LabelFalse, E))
2882 if (!this->jump(LabelEnd, E))
2884 this->emitLabel(LabelFalse);
2888 this->fallthrough(LabelEnd);
2889 this->emitLabel(LabelEnd);
2892 return this->emitPopIgnoreDiags(E);
2896template <
class Emitter>
2902 unsigned StringIndex =
P.createGlobalString(E);
2903 return this->emitGetPtrGlobal(StringIndex, E);
2908 Ctx.getASTContext().getAsConstantArrayType(E->
getType());
2909 assert(CAT &&
"a string literal that's not a constant array?");
2914 unsigned N = std::min(ArraySize, E->
getLength());
2917 for (
unsigned I = 0; I != N; ++I) {
2920 if (CharWidth == 1) {
2921 this->emitConstSint8(CodeUnit, E);
2922 this->emitInitElemSint8(I, E);
2923 }
else if (CharWidth == 2) {
2924 this->emitConstUint16(CodeUnit, E);
2925 this->emitInitElemUint16(I, E);
2926 }
else if (CharWidth == 4) {
2927 this->emitConstUint32(CodeUnit, E);
2928 this->emitInitElemUint32(I, E);
2930 llvm_unreachable(
"unsupported character width");
2935 for (
unsigned I = N; I != ArraySize; ++I) {
2936 if (CharWidth == 1) {
2937 this->emitConstSint8(0, E);
2938 this->emitInitElemSint8(I, E);
2939 }
else if (CharWidth == 2) {
2940 this->emitConstUint16(0, E);
2941 this->emitInitElemUint16(I, E);
2942 }
else if (CharWidth == 4) {
2943 this->emitConstUint32(0, E);
2944 this->emitInitElemUint32(I, E);
2946 llvm_unreachable(
"unsupported character width");
2953template <
class Emitter>
2957 return this->emitDummyPtr(E, E);
2960template <
class Emitter>
2962 auto &A =
Ctx.getASTContext();
2971template <
class Emitter>
2979 auto &A =
Ctx.getASTContext();
2983 APInt Size(A.getTypeSize(A.getSizeType()), ResultStr.size() + 1);
2984 QualType ArrayTy = A.getConstantArrayType(CharTy, Size,
nullptr,
2991 unsigned StringIndex =
P.createGlobalString(SL);
2992 return this->emitGetPtrGlobal(StringIndex, E);
2995template <
class Emitter>
2999 return this->emitConst(E->
getValue(), E);
3002template <
class Emitter>
3028 if (!this->emitSetLocal(*RT, TempOffset, E))
3034 if (!this->emitLoad(LHST, E))
3038 if (!this->emitPrimCast(LHST,
classifyPrim(LHSComputationType),
3039 LHSComputationType, E))
3043 if (!this->emitGetLocal(*RT, TempOffset, E))
3048 if (!this->emitAddf(getFPOptions(E), E))
3052 if (!this->emitSubf(getFPOptions(E), E))
3056 if (!this->emitMulf(getFPOptions(E), E))
3060 if (!this->emitDivf(getFPOptions(E), E))
3071 return this->emitStorePop(LHST, E);
3072 return this->emitStore(LHST, E);
3075template <
class Emitter>
3084 if (Op != BO_AddAssign && Op != BO_SubAssign)
3093 if (!this->emitLoad(*
LT, LHS))
3099 if (Op == BO_AddAssign) {
3100 if (!this->emitAddOffset(*RT, E))
3103 if (!this->emitSubOffset(*RT, E))
3108 return this->emitStorePopPtr(E);
3109 return this->emitStorePtr(E);
3112template <
class Emitter>
3125 if (!
Ctx.getLangOpts().CPlusPlus14)
3126 return this->
visit(RHS) && this->
visit(LHS) && this->emitError(E);
3128 if (!
LT || !RT || !ResultT || !LHSComputationT)
3153 if (!this->emitSetLocal(*RT, TempOffset, E))
3160 if (!this->emitLoad(*
LT, E))
3162 if (
LT != LHSComputationT &&
3168 if (!this->emitGetLocal(*RT, TempOffset, E))
3174 if (!this->emitAdd(*LHSComputationT, E))
3178 if (!this->emitSub(*LHSComputationT, E))
3182 if (!this->emitMul(*LHSComputationT, E))
3186 if (!this->emitDiv(*LHSComputationT, E))
3190 if (!this->emitRem(*LHSComputationT, E))
3194 if (!this->emitShl(*LHSComputationT, *RT, E))
3198 if (!this->emitShr(*LHSComputationT, *RT, E))
3202 if (!this->emitBitAnd(*LHSComputationT, E))
3206 if (!this->emitBitXor(*LHSComputationT, E))
3210 if (!this->emitBitOr(*LHSComputationT, E))
3214 llvm_unreachable(
"Unimplemented compound assign operator");
3218 if (ResultT != LHSComputationT &&
3219 !this->emitIntegralCast(*LHSComputationT, *ResultT, E->
getType(), E))
3225 return this->emitStoreBitFieldPop(*ResultT, E);
3226 return this->emitStorePop(*ResultT, E);
3229 return this->emitStoreBitField(*ResultT, E);
3230 return this->emitStore(*ResultT, E);
3233template <
class Emitter>
3241template <
class Emitter>
3256 if (!
Ctx.getLangOpts().CPlusPlus11)
3263 for (
const Expr *LHS : CommaLHSs) {
3285 if (!this->
visit(Inner))
3290 if (!this->emitInitGlobalTemp(*InnerT, *GlobalIndex, TempDecl, E))
3293 if (!this->emitInitGlobal(*InnerT, *GlobalIndex, E))
3296 return this->emitGetPtrGlobal(*GlobalIndex, E);
3299 if (!this->checkLiteralType(Inner))
3302 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3308 return this->emitInitGlobalTempComp(TempDecl, E);
3321 unsigned LocalIndex =
3323 if (!this->VarScope->LocalsAlwaysEnabled &&
3324 !this->emitEnableLocal(LocalIndex, E))
3327 if (!this->
visit(Inner))
3329 if (!this->emitSetLocal(*InnerT, LocalIndex, E))
3332 return this->emitGetPtrLocal(LocalIndex, E);
3335 if (!this->checkLiteralType(Inner))
3342 if (!this->VarScope->LocalsAlwaysEnabled &&
3343 !this->emitEnableLocal(*LocalIndex, E))
3346 if (!this->emitGetPtrLocal(*LocalIndex, E))
3353template <
class Emitter>
3364 if (!this->
visit(SubExpr))
3368 return this->emitPopPtr(E);
3372template <
class Emitter>
3393 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3398 if (
P.isGlobalInitialized(*GlobalIndex))
3404 return this->emitInitGlobal(*T, *GlobalIndex, E);
3416 unsigned LocalIndex;
3420 LocalIndex = *MaybeIndex;
3424 if (!this->emitGetPtrLocal(LocalIndex, E))
3428 return this->
visit(
Init) && this->emitInit(*T, E);
3432template <
class Emitter>
3445template <
class Emitter>
3449 return this->emitConst(E->
getValue(), E);
3452template <
class Emitter>
3465 for (
const Record::Field &F : R->fields()) {
3467 if (!
Init ||
Init->containsErrors())
3475 if (!this->emitInitField(*T, F.Offset, E))
3478 if (!this->emitGetPtrField(F.Offset, E))
3489template <
class Emitter>
3496 return this->emitGetPtrGlobal(StringIndex, E);
3502template <
class Emitter>
3507 return this->emitInvalid(E);
3510template <
class Emitter>
3536 if (PointeeToT && PointeeFromT) {
3555 bool Fatal = (ToT != FromT);
3562template <
class Emitter>
3565 if (!
Ctx.getLangOpts().CPlusPlus20) {
3573 if (!this->emitCheckDynamicCast(E))
3577 return this->emitPopPtr(E);
3581template <
class Emitter>
3587 return this->emitConstBool(E->
getValue(), E);
3590template <
class Emitter>
3595 if (T->isRecordType()) {
3609 if (!this->emitGetPtrLocal(*LocalIndex, E))
3617 T->getAsCXXRecordDecl()))
3627 if (!this->visitZeroRecordInitializer(R, E))
3639 assert(
Ctx.getASTContext().hasSameUnqualifiedType(E->
getType(),
3641 if (
const auto *ME = dyn_cast<MaterializeTemporaryExpr>(SrcObj)) {
3642 if (!this->emitCheckFunctionDecl(Ctor, E))
3653 assert(
Func->hasThisPointer());
3654 assert(!
Func->hasRVO());
3658 if (!this->emitDupPtr(E))
3662 for (
const auto *Arg : E->
arguments()) {
3663 if (!this->
visit(Arg))
3667 if (
Func->isVariadic()) {
3668 uint32_t VarArgSize = 0;
3669 unsigned NumParams =
Func->getNumWrittenParams();
3670 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I) {
3674 if (!this->emitCallVar(
Func, VarArgSize, E))
3677 if (!this->emitCall(
Func, 0, E)) {
3682 (void)this->emitPopPtr(E);
3688 return this->emitPopPtr(E);
3692 if (T->isArrayType()) {
3697 if (!this->emitDupPtr(E))
3701 initArrayDimension = [&](
QualType T) ->
bool {
3702 if (!T->isArrayType()) {
3704 for (
const auto *Arg : E->
arguments()) {
3705 if (!this->
visit(Arg))
3709 return this->emitCall(
Func, 0, E);
3713 Ctx.getASTContext().getAsConstantArrayType(T);
3718 for (
size_t I = 0; I != NumElems; ++I) {
3719 if (!this->emitConstUint64(I, E))
3721 if (!this->emitArrayElemPtrUint64(E))
3723 if (!initArrayDimension(ElemTy))
3726 return this->emitPopPtr(E);
3729 return initArrayDimension(E->
getType());
3735template <
class Emitter>
3745 assert(Val.
isInt());
3747 return this->emitConst(I, E);
3754 if (
const Expr *LValueExpr =
Base.dyn_cast<
const Expr *>())
3755 return this->
visit(LValueExpr);
3770 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
3774 const APValue &
V = UGCD->getValue();
3775 for (
unsigned I = 0, N = R->getNumFields(); I != N; ++I) {
3776 const Record::Field *F = R->getField(I);
3777 const APValue &FieldValue =
V.getStructField(I);
3783 if (!this->emitInitField(FieldT, F->Offset, E))
3791template <
class Emitter>
3797 for (
unsigned I = 0; I != N; ++I) {
3804 if (!this->
discard(ArrayIndexExpr))
3809 if (!this->
visit(ArrayIndexExpr))
3813 if (!this->emitCast(IndexT,
PT_Sint64, E))
3823 return this->emitOffsetOf(T, E, E);
3826template <
class Emitter>
3835 return this->visitZeroInitializer(*T, Ty, E);
3842 if (!this->emitGetPtrLocal(*LocalIndex, E))
3850 ElemQT = CT->getElementType();
3853 NumElems = VT->getNumElements();
3854 ElemQT = VT->getElementType();
3860 for (
unsigned I = 0, N = NumElems; I != N; ++I) {
3861 if (!this->visitZeroInitializer(ElemT, ElemQT, E))
3863 if (!this->emitInitElem(ElemT, I, E))
3872template <
class Emitter>
3877template <
class Emitter>
3883template <
class Emitter>
3888template <
class Emitter>
3893 return this->emitConst(E->
getValue(), E);
3896template <
class Emitter>
3901 "Trivial CXXInheritedCtorInitExpr, implement. (possible?)");
3915 unsigned ParamIndex = 0;
3919 if (!this->emitGetParam(PT, ParamIndex, E))
3924 return this->emitCall(F, 0, E);
3929template <
class Emitter>
3937 const Expr *PlacementDest =
nullptr;
3938 bool IsNoThrow =
false;
3943 if (PlacementArgs != 0) {
3952 if (PlacementArgs == 1) {
3960 if (!this->emitInvalidNewDeleteExpr(E, E))
3965 if (OperatorNew->isReservedGlobalPlacementOperator())
3966 PlacementDest = Arg1;
3970 return this->emitInvalid(E);
3972 }
else if (!OperatorNew
3973 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
3974 return this->emitInvalidNewDeleteExpr(E, E);
3977 if (!PlacementDest) {
3982 Desc =
P.createDescriptor(E, *ElemT,
nullptr,
3985 Desc =
P.createDescriptor(
3988 false,
false,
false,
3994 std::optional<const Expr *> ArraySizeExpr = E->
getArraySize();
3998 const Expr *Stripped = *ArraySizeExpr;
3999 for (;
auto *ICE = dyn_cast<ImplicitCastExpr>(Stripped);
4000 Stripped = ICE->getSubExpr())
4001 if (ICE->getCastKind() != CK_NoOp &&
4002 ICE->getCastKind() != CK_IntegralCast)
4010 if (!this->
visit(Stripped))
4012 if (!this->emitSetLocal(
SizeT, ArrayLen, E))
4015 if (PlacementDest) {
4016 if (!this->
visit(PlacementDest))
4018 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4020 if (!this->emitCheckNewTypeMismatchArray(
SizeT, E, E))
4023 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4028 if (!this->emitAllocN(
SizeT, *ElemT, E, IsNoThrow, E))
4032 if (!this->emitAllocCN(
SizeT, Desc, IsNoThrow, E))
4039 size_t StaticInitElems = 0;
4040 const Expr *DynamicInit =
nullptr;
4044 Ctx.getASTContext().getAsConstantArrayType(InitType)) {
4045 StaticInitElems = CAT->getZExtSize();
4050 if (
const auto *ILE = dyn_cast<InitListExpr>(
Init)) {
4051 if (ILE->hasArrayFiller())
4052 DynamicInit = ILE->getArrayFiller();
4071 const Function *CtorFunc =
nullptr;
4072 if (
const auto *CE = dyn_cast<CXXConstructExpr>(
Init)) {
4076 }
else if (!DynamicInit && !ElemT)
4079 LabelTy EndLabel = this->getLabel();
4080 LabelTy StartLabel = this->getLabel();
4085 if (!this->emitDupPtr(E))
4087 if (!this->emitNullPtr(0,
nullptr, E))
4089 if (!this->emitEQPtr(E))
4091 if (!this->jumpTrue(EndLabel, E))
4098 if (!this->emitConst(StaticInitElems,
SizeT, E))
4100 if (!this->emitSetLocal(
SizeT, Iter, E))
4103 this->fallthrough(StartLabel);
4104 this->emitLabel(StartLabel);
4106 if (!this->emitGetLocal(
SizeT, Iter, E))
4108 if (!this->emitGetLocal(
SizeT, ArrayLen, E))
4110 if (!this->emitLT(
SizeT, E))
4112 if (!this->jumpFalse(EndLabel, E))
4116 if (!this->emitGetLocal(
SizeT, Iter, E))
4118 if (!this->emitArrayElemPtr(
SizeT, E))
4121 if (isa_and_nonnull<ImplicitValueInitExpr>(DynamicInit) &&
4126 if (!this->visitZeroInitializer(InitT, ElemType, E))
4128 if (!this->emitStorePop(InitT, E))
4130 }
else if (DynamicInit) {
4132 if (!this->
visit(DynamicInit))
4134 if (!this->emitStorePop(*InitT, E))
4141 if (!this->visitZeroInitializer(
4145 if (!this->emitStorePop(*ElemT, E))
4149 if (!this->emitCall(CtorFunc, 0, E))
4154 if (!this->emitGetPtrLocal(Iter, E))
4156 if (!this->emitIncPop(
SizeT,
false, E))
4159 if (!this->jump(StartLabel, E))
4162 this->fallthrough(EndLabel);
4163 this->emitLabel(EndLabel);
4167 if (PlacementDest) {
4168 if (!this->
visit(PlacementDest))
4170 if (!this->emitCheckNewTypeMismatch(E, E))
4175 if (!this->emitAlloc(Desc, E))
4184 if (!this->emitInit(*ElemT, E))
4195 return this->emitPopPtr(E);
4200template <
class Emitter>
4206 if (!OperatorDelete->isUsableAsGlobalAllocationFunctionInConstantEvaluation())
4207 return this->emitInvalidNewDeleteExpr(E, E);
4216template <
class Emitter>
4222 if (
const Function *F =
Ctx.getOrCreateObjCBlock(E))
4227 return this->emitGetFnPtr(
Func, E);
4230template <
class Emitter>
4234 auto canonType = [](
const Type *T) {
4235 return T->getCanonicalTypeUnqualified().getTypePtr();
4243 return this->emitGetTypeid(
4247 return this->emitGetTypeid(
4256 if (!
Ctx.getLangOpts().CPlusPlus20 && !this->emitDiagTypeid(E))
4262 if (!this->emitGetTypeidPtr(TypeInfoType, E))
4265 return this->emitPopPtr(E);
4269template <
class Emitter>
4273 return this->emitDummyPtr(E, E);
4274 return this->emitError(E);
4277template <
class Emitter>
4280 return this->emitDummyPtr(E, E);
4281 return this->emitError(E);
4284template <
class Emitter>
4286 assert(
Ctx.getLangOpts().CPlusPlus);
4287 return this->emitConstBool(E->
getValue(), E);
4290template <
class Emitter>
4302 return this->emitDummyPtr(GuidDecl, E);
4307 if (!this->emitGetPtrGlobal(*GlobalIndex, E))
4316 assert(
V.isStruct());
4317 assert(
V.getStructNumBases() == 0);
4321 return this->emitFinishInit(E);
4324template <
class Emitter>
4334template <
class Emitter>
4343template <
class Emitter>
4349template <
class Emitter>
4353 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(SemE)) {
4357 if (OVE->isUnique())
4373template <
class Emitter>
4378template <
class Emitter>
4380 return this->emitError(E);
4383template <
class Emitter>
4389 return this->emitDummyPtr(E, E);
4392template <
class Emitter>
4393bool Compiler<Emitter>::emitVectorConversion(
const Expr *Src,
const Expr *E) {
4398 QualType ElemType = VT->getElementType();
4399 PrimType ElemT = classifyPrim(ElemType);
4401 PrimType SrcElemT = classifyVectorElementType(SrcType);
4407 if (!this->emitGetPtrLocal(*LocalIndex, E))
4411 unsigned SrcOffset =
4412 this->allocateLocalPrimitive(Src,
PT_Ptr,
true);
4413 if (!this->visit(Src))
4415 if (!this->emitSetLocal(
PT_Ptr, SrcOffset, E))
4418 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
4419 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
4421 if (!this->emitArrayElemPop(SrcElemT, I, E))
4425 if (SrcElemT != ElemT) {
4426 if (!this->emitPrimCast(SrcElemT, ElemT, ElemType, E))
4428 }
else if (ElemType->isFloatingType() && SrcType != ElemType) {
4429 const auto *TargetSemantics = &Ctx.getFloatSemantics(ElemType);
4433 if (!this->emitInitElem(ElemT, I, E))
4439template <
class Emitter>
4441 return emitVectorConversion(E->
getSrcExpr(), E);
4444template <
class Emitter>
4448 return this->emitInvalid(E);
4457 assert(NumOutputElems > 0);
4463 if (!this->emitGetPtrLocal(*LocalIndex, E))
4468 unsigned VectorOffsets[2];
4469 for (
unsigned I = 0; I != 2; ++I) {
4472 if (!this->
visit(Vecs[I]))
4474 if (!this->emitSetLocal(
PT_Ptr, VectorOffsets[I], E))
4477 for (
unsigned I = 0; I != NumOutputElems; ++I) {
4479 assert(ShuffleIndex >= -1);
4480 if (ShuffleIndex == -1)
4481 return this->emitInvalidShuffleVectorIndex(I, E);
4483 assert(ShuffleIndex < (NumInputElems * 2));
4484 if (!this->emitGetLocal(
PT_Ptr,
4485 VectorOffsets[ShuffleIndex >= NumInputElems], E))
4487 unsigned InputVectorIndex = ShuffleIndex.getZExtValue() % NumInputElems;
4488 if (!this->emitArrayElemPop(ElemT, InputVectorIndex, E))
4491 if (!this->emitInitElem(ElemT, I, E))
4496 return this->emitPopPtr(E);
4501template <
class Emitter>
4506 Base->getType()->isVectorType() ||
4512 if (Indices.size() == 1) {
4517 if (!this->emitConstUint32(Indices[0], E))
4519 return this->emitArrayElemPtrPop(
PT_Uint32, E);
4529 if (!this->emitSetLocal(
PT_Ptr, BaseOffset, E))
4537 if (!this->emitGetPtrLocal(*ResultIndex, E))
4545 uint32_t DstIndex = 0;
4546 for (uint32_t I : Indices) {
4547 if (!this->emitGetLocal(
PT_Ptr, BaseOffset, E))
4549 if (!this->emitArrayElemPop(ElemT, I, E))
4551 if (!this->emitInitElem(ElemT, DstIndex, E))
4561template <
class Emitter>
4565 return this->
discard(SubExpr) && this->emitInvalid(E);
4571 return this->emitDummyPtr(E, E);
4574template <
class Emitter>
4579 Ctx.getASTContext().getAsConstantArrayType(SubExpr->
getType());
4584 if (!this->
visit(SubExpr))
4586 if (!this->emitConstUint8(0, E))
4588 if (!this->emitArrayElemPtrPopUint8(E))
4590 if (!this->emitInitFieldPtr(R->getField(0u)->Offset, E))
4595 if (!this->emitConst(
ArrayType->getSize(), SecondFieldT, E))
4597 return this->emitInitField(SecondFieldT, R->getField(1u)->Offset, E);
4599 assert(SecondFieldT ==
PT_Ptr);
4601 if (!this->emitGetFieldPtr(R->getField(0u)->Offset, E))
4603 if (!this->emitExpandPtr(E))
4607 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
4609 return this->emitInitFieldPtr(R->getField(1u)->Offset, E);
4612template <
class Emitter>
4627 if (
const Expr *ResultExpr = dyn_cast<Expr>(S))
4629 return this->emitUnsupported(E);
4638 return this->
Visit(E);
4645 return this->
Visit(E);
4649 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4652 if (
const auto *CE = dyn_cast<CastExpr>(E);
4654 (CE->getCastKind() == CK_DerivedToBase || CE->getCastKind() == CK_NoOp))
4661 if (
const auto *PE = dyn_cast<ParenExpr>(E))
4664 if (
const auto *CE = dyn_cast<CastExpr>(E);
4665 CE && (CE->getCastKind() == CK_DerivedToBase ||
4666 CE->getCastKind() == CK_UncheckedDerivedToBase ||
4667 CE->getCastKind() == CK_NoOp))
4687 if (!this->emitGetPtrLocal(*LocalIndex, E))
4697 return this->
Visit(E);
4700template <
class Emitter>
4706 return this->
Visit(E) && this->emitFinishInit(E);
4709template <
class Emitter>
4715 return this->
Visit(E) && this->emitFinishInitPop(E);
4721 return this->
Visit(E);
4732 if (!this->
visit(E))
4734 return this->emitComplexBoolCast(E);
4739 if (!this->
visit(E))
4747 return this->emitIsNonNullPtr(E);
4751 return this->emitCastFloatingIntegralBool(getFPOptions(E), E);
4754 return this->emitCast(*T,
PT_Bool, E);
4757template <
class Emitter>
4761 QT = AT->getValueType();
4765 return this->emitZeroBool(E);
4767 return this->emitZeroSint8(E);
4769 return this->emitZeroUint8(E);
4771 return this->emitZeroSint16(E);
4773 return this->emitZeroUint16(E);
4775 return this->emitZeroSint32(E);
4777 return this->emitZeroUint32(E);
4779 return this->emitZeroSint64(E);
4781 return this->emitZeroUint64(E);
4783 return this->emitZeroIntAP(Ctx.getBitWidth(QT), E);
4785 return this->emitZeroIntAPS(Ctx.getBitWidth(QT), E);
4787 return this->emitNullPtr(Ctx.getASTContext().getTargetNullPointerValue(QT),
4790 return this->emitNullMemberPtr(0,
nullptr, E);
4792 APFloat F = APFloat::getZero(Ctx.getFloatSemantics(QT));
4793 return this->emitFloat(F, E);
4796 auto Sem = Ctx.getASTContext().getFixedPointSemantics(QT);
4800 llvm_unreachable(
"unknown primitive type");
4803template <
class Emitter>
4804bool Compiler<Emitter>::visitZeroRecordInitializer(
const Record *R,
4809 for (
const Record::Field &Field :
R->fields()) {
4810 if (Field.isUnnamedBitField())
4817 if (!this->visitZeroInitializer(T, QT, E))
4820 if (!this->emitInitFieldActivate(T, Field.Offset, E))
4824 if (!this->emitInitField(T, Field.Offset, E))
4829 if (!this->emitGetPtrField(Field.Offset, E))
4835 for (uint32_t I = 0, N = D->
getNumElems(); I != N; ++I) {
4836 if (!this->visitZeroInitializer(T, ET, E))
4838 if (!this->emitInitElem(T, I, E))
4843 if (!this->visitZeroArrayInitializer(D->
getType(), E))
4846 if (!this->visitZeroRecordInitializer(D->
ElemRecord, E))
4854 if (!this->emitFinishInitActivatePop(E))
4858 if (!this->emitFinishInitPop(E))
4862 for (
const Record::Base &B :
R->bases()) {
4863 if (!this->emitGetPtrBase(B.Offset, E))
4865 if (!this->visitZeroRecordInitializer(B.R, E))
4867 if (!this->emitFinishInitPop(E))
4876template <
class Emitter>
4877bool Compiler<Emitter>::visitZeroArrayInitializer(
QualType T,
const Expr *E) {
4878 assert(T->isArrayType() || T->isAnyComplexType() || T->isVectorType());
4884 for (
size_t I = 0; I != NumElems; ++I) {
4885 if (!this->visitZeroInitializer(*ElemT, ElemType, E))
4887 if (!this->emitInitElem(*ElemT, I, E))
4893 const Record *
R = getRecord(ElemType);
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->visitZeroRecordInitializer(R, E))
4904 if (!this->emitPopPtr(E))
4910 for (
size_t I = 0; I != NumElems; ++I) {
4911 if (!this->emitConstUint32(I, E))
4913 if (!this->emitArrayElemPtr(
PT_Uint32, E))
4915 if (!this->visitZeroArrayInitializer(ElemType, E))
4917 if (!this->emitPopPtr(E))
4926template <
class Emitter>
4927bool Compiler<Emitter>::visitAssignment(
const Expr *LHS,
const Expr *RHS,
4929 if (!canClassify(E->
getType()))
4932 if (!this->visit(RHS))
4934 if (!this->visit(LHS))
4941 if (!Ctx.getLangOpts().CPlusPlus && !this->emitInvalid(E))
4945 bool Activates = refersToUnion(LHS);
4948 if (!this->emitFlip(
PT_Ptr, RHT, E))
4951 if (DiscardResult) {
4952 if (BitField && Activates)
4953 return this->emitStoreBitFieldActivatePop(RHT, E);
4955 return this->emitStoreBitFieldPop(RHT, E);
4957 return this->emitStoreActivatePop(RHT, E);
4959 return this->emitStorePop(RHT, E);
4962 auto maybeLoad = [&](
bool Result) ->
bool {
4968 return this->emitLoadPop(RHT, E);
4972 if (BitField && Activates)
4973 return maybeLoad(this->emitStoreBitFieldActivate(RHT, E));
4975 return maybeLoad(this->emitStoreBitField(RHT, E));
4977 return maybeLoad(this->emitStoreActivate(RHT, E));
4979 return maybeLoad(this->emitStore(RHT, E));
4982template <
class Emitter>
4983template <
typename T>
4987 return this->emitConstSint8(
Value, E);
4989 return this->emitConstUint8(
Value, E);
4991 return this->emitConstSint16(
Value, E);
4993 return this->emitConstUint16(
Value, E);
4995 return this->emitConstSint32(
Value, E);
4997 return this->emitConstUint32(
Value, E);
4999 return this->emitConstSint64(
Value, E);
5001 return this->emitConstUint64(
Value, E);
5003 return this->emitConstBool(
Value, E);
5010 llvm_unreachable(
"Invalid integral type");
5013 llvm_unreachable(
"unknown primitive type");
5016template <
class Emitter>
5017template <
typename T>
5018bool Compiler<Emitter>::emitConst(T
Value,
const Expr *E) {
5019 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5022template <
class Emitter>
5026 return this->emitConstIntAPS(
Value, E);
5028 return this->emitConstIntAP(
Value, E);
5030 if (
Value.isSigned())
5031 return this->emitConst(
Value.getSExtValue(), Ty, E);
5032 return this->emitConst(
Value.getZExtValue(), Ty, E);
5035template <
class Emitter>
5039 return this->emitConstIntAPS(
Value, E);
5041 return this->emitConstIntAP(
Value, E);
5044 return this->emitConst(
Value.getSExtValue(), Ty, E);
5045 return this->emitConst(
Value.getZExtValue(), Ty, E);
5048template <
class Emitter>
5049bool Compiler<Emitter>::emitConst(
const APSInt &
Value,
const Expr *E) {
5050 return this->emitConst(
Value, classifyPrim(E->
getType()), E);
5053template <
class Emitter>
5066 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>()))
5067 Locals.insert({VD, Local});
5068 VarScope->addForScopeKind(Local, SC);
5069 return Local.Offset;
5072template <
class Emitter>
5077 bool IsTemporary =
false;
5078 if (
auto *VD = dyn_cast_if_present<ValueDecl>(Src.dyn_cast<
const Decl *>())) {
5081 if (
const auto *VarD = dyn_cast<VarDecl>(VD))
5082 Init = VarD->getInit();
5084 if (
auto *E = Src.dyn_cast<
const Expr *>()) {
5095 return std::nullopt;
5100 Locals.insert({Key, Local});
5101 VarScope->addForScopeKind(Local, SC);
5102 return Local.Offset;
5105template <
class Emitter>
5115 return std::nullopt;
5125 return Local.Offset;
5128template <
class Emitter>
5130 if (
const PointerType *PT = dyn_cast<PointerType>(Ty))
5131 return PT->getPointeeType()->getAsCanonical<RecordType>();
5137 return getRecord(RecordTy->getDecl()->getDefinitionOrSelf());
5141template <
class Emitter>
5143 return P.getOrCreateRecord(RD);
5146template <
class Emitter>
5148 return Ctx.getOrCreateFunction(FD);
5151template <
class Emitter>
5155 auto maybeDestroyLocals = [&]() ->
bool {
5156 if (DestroyToplevelScope)
5157 return RootScope.
destroyLocals() && this->emitCheckAllocations(E);
5158 return this->emitCheckAllocations(E);
5165 return this->emitRetVoid(E) && maybeDestroyLocals();
5173 return this->emitRet(*T, E) && maybeDestroyLocals();
5181 if (!this->emitGetPtrLocal(*LocalOffset, E))
5189 return this->emitRetValue(E) && maybeDestroyLocals();
5192 return maybeDestroyLocals() &&
false;
5195template <
class Emitter>
5197 bool DestroyToplevelScope) {
5201 return this->
visitExpr(E, DestroyToplevelScope);
5204template <
class Emitter>
5216 if (
auto GlobalIndex =
P.getGlobal(VD)) {
5217 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5231template <
class Emitter>
5233 bool ConstantContext) {
5236 if (!ConstantContext) {
5250 auto GlobalIndex =
P.getGlobal(VD);
5251 assert(GlobalIndex);
5253 if (!this->emitGetGlobalUnchecked(*VarT, *GlobalIndex, VD))
5256 if (!this->emitGetPtrGlobal(*GlobalIndex, VD))
5260 auto Local =
Locals.find(VD);
5261 assert(Local !=
Locals.end());
5263 if (!this->emitGetLocal(*VarT, Local->second.Offset, VD))
5266 if (!this->emitGetPtrLocal(Local->second.Offset, VD))
5276 auto GlobalIndex =
P.getGlobal(VD);
5277 assert(GlobalIndex);
5278 Block *GlobalBlock =
P.getGlobal(*GlobalIndex);
5287 return VDScope.
destroyLocals() && this->emitCheckAllocations(VD);
5290template <
class Emitter>
5301 if (!this->isActive())
5306 if (
Init &&
Init->isValueDependent())
5310 auto checkDecl = [&]() ->
bool {
5312 return !NeedsOp || this->emitCheckDecl(VD, VD);
5319 if (!
P.getGlobal(*GlobalIndex)->isInitialized())
5322 if (
P.isGlobalInitialized(*GlobalIndex))
5326 }
else if ((GlobalIndex =
5341 return this->emitInitGlobal(*VarT, *GlobalIndex, VD);
5344 if (!this->emitGetPtrGlobal(*GlobalIndex,
Init))
5347 if (!this->emitStartInit(
Init))
5353 if (!this->emitEndInit(
Init))
5356 return this->emitFinishInitGlobal(
Init);
5366 if (!
Init ||
Init->getType()->isVoidType())
5375 return this->emitSetLocal(*VarT, Offset, VD) &&
Scope.destroyLocals();
5387 if (!this->emitCheckRefInit(
Init))
5391 return this->emitSetLocal(*VarT, Offset, VD);
5399 if (!this->emitGetPtrLocal(*Offset,
Init))
5407template <
class Emitter>
5422 Locals.insert({VD, Local});
5425 if (!this->emitGetPtrLocal(Local.Offset, VD))
5429 return this->emitDestructionPop(D, VD);
5432template <
class Emitter>
5437 return this->emitConst(Val.
getInt(), ValType, E);
5440 return this->emitFloat(F, E);
5445 if (!this->emitGetMemberPtr(MemberDecl, E))
5451 if (!this->emitCopyMemberPtrPath(PathEntry, IsDerived, E))
5457 return this->emitNullMemberPtr(0,
nullptr, E);
5462 return this->emitNull(ValType, 0,
nullptr, E);
5467 if (
const Expr *BaseExpr =
Base.dyn_cast<
const Expr *>())
5468 return this->
visit(BaseExpr);
5473 QualType EntryType = VD->getType();
5474 for (
auto &Entry : Path) {
5476 uint64_t Index = Entry.getAsArrayIndex();
5479 if (!this->emitConst(Index,
PT_Uint64, E))
5481 if (!this->emitArrayElemPtrPop(
PT_Uint64, E))
5483 EntryType = ElemType;
5493 const Decl *BaseOrMember = Entry.getAsBaseOrMember().getPointer();
5494 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
5496 if (!this->emitGetPtrFieldPop(EntryOffset, E))
5498 EntryType = FD->getType();
5502 if (!this->emitGetPtrBasePop(BaseOffset,
false, E))
5504 EntryType =
Ctx.getASTContext().getCanonicalTagType(
Base);
5516template <
class Emitter>
5524 const Record::Field *RF = R->getField(I);
5525 QualType FieldType = RF->Decl->getType();
5531 if (!this->emitInitField(*PT, RF->Offset, E))
5534 if (!this->emitGetPtrField(RF->Offset, E))
5538 if (!this->emitFinishInitPop(E))
5546 const Record::Base *RB = R->getBase(I);
5547 QualType BaseType =
Ctx.getASTContext().getCanonicalTagType(RB->Decl);
5549 if (!this->emitGetPtrBase(RB->Offset, E))
5553 if (!this->emitFinishInitPop(E))
5566 const Record::Field *RF = R->getField(UnionField);
5567 QualType FieldType = RF->Decl->getType();
5572 if (RF->isBitField())
5573 return this->emitInitBitFieldActivate(*PT, RF->Offset, RF->bitWidth(),
5575 return this->emitInitFieldActivate(*PT, RF->Offset, E);
5578 if (!this->emitGetPtrField(RF->Offset, E))
5580 if (!this->emitActivate(E))
5584 return this->emitPopPtr(E);
5588 const auto *ArrType = T->getAsArrayTypeUnsafe();
5589 QualType ElemType = ArrType->getElementType();
5592 for (
unsigned A = 0, AN = Val.
getArraySize(); A != AN; ++A) {
5593 const APValue &Elem = A >= InitializedElems
5600 if (!this->emitInitElem(*ElemT, A, E))
5603 if (!this->emitConstUint32(A, E))
5605 if (!this->emitArrayElemPtrUint32(E))
5609 if (!this->emitPopPtr(E))
5620template <
class Emitter>
5622 unsigned BuiltinID) {
5623 if (BuiltinID == Builtin::BI__builtin_constant_p) {
5628 return this->emitConst(0, E);
5631 if (!this->emitStartSpeculation(E))
5633 LabelTy EndLabel = this->getLabel();
5634 if (!this->speculate(E, EndLabel))
5636 if (!this->emitEndSpeculation(E))
5638 this->fallthrough(EndLabel);
5646 if (BuiltinID == Builtin::BI__builtin___CFStringMakeConstantString ||
5647 BuiltinID == Builtin::BI__builtin___NSStringMakeConstantString ||
5648 BuiltinID == Builtin::BI__builtin_ptrauth_sign_constant ||
5649 BuiltinID == Builtin::BI__builtin_function_start) {
5652 return this->emitDummyPtr(E, E);
5663 if (!this->emitGetPtrLocal(*LocalIndex, E))
5668 switch (BuiltinID) {
5669 case Builtin::BI__builtin_object_size:
5670 case Builtin::BI__builtin_dynamic_object_size: {
5674 if (!this->
visit(Arg0))
5685 case Builtin::BI__assume:
5686 case Builtin::BI__builtin_assume:
5689 case Builtin::BI__atomic_is_lock_free:
5690 case Builtin::BI__atomic_always_lock_free: {
5701 for (
const auto *Arg : E->
arguments()) {
5702 if (!this->
visit(Arg))
5708 if (!this->emitCallBI(E, BuiltinID, E))
5717template <
class Emitter>
5738 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(FuncDecl);
5739 DD && DD->isTrivial()) {
5741 if (!this->
visit(MemberCall->getImplicitObjectArgument()))
5743 return this->emitCheckDestruction(E) && this->emitEndLifetime(E) &&
5744 this->emitPopPtr(E);
5752 bool HasRVO = !
ReturnType->isVoidType() && !T;
5760 if (!this->emitGetPtrLocal(*LocalIndex, E))
5768 if (!this->emitGetPtrLocal(*LocalIndex, E))
5772 if (!this->emitDupPtr(E))
5779 bool IsAssignmentOperatorCall =
false;
5780 bool ActivateLHS =
false;
5781 if (
const auto *OCE = dyn_cast<CXXOperatorCallExpr>(E);
5782 OCE && OCE->isAssignmentOp()) {
5786 assert(Args.size() == 2);
5787 const CXXRecordDecl *LHSRecord = Args[0]->getType()->getAsCXXRecordDecl();
5789 IsAssignmentOperatorCall =
true;
5790 std::reverse(Args.begin(), Args.end());
5796 if (
const auto *MD = dyn_cast_if_present<CXXMethodDecl>(FuncDecl);
5797 MD && MD->isStatic()) {
5801 Args.erase(Args.begin());
5805 bool Devirtualized =
false;
5808 if (
const auto *MC = dyn_cast<CXXMemberCallExpr>(E)) {
5816 if (!this->
visit(Callee))
5818 if (!this->emitSetLocal(
PT_MemberPtr, *CalleeOffset, E))
5820 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5822 if (!this->emitGetMemberPtrBase(E))
5825 const auto *InstancePtr = MC->getImplicitObjectArgument();
5832 Stripped->getType()->getPointeeType()->getAsCXXRecordDecl());
5833 Devirtualized =
true;
5834 if (!this->
visit(Stripped))
5837 if (!this->
visit(InstancePtr))
5841 if (!this->
visit(InstancePtr))
5845 }
else if (
const auto *PD =
5846 dyn_cast<CXXPseudoDestructorExpr>(E->
getCallee())) {
5847 if (!this->emitCheckPseudoDtor(E))
5855 return this->emitEndLifetimePop(E);
5856 }
else if (!FuncDecl) {
5860 if (!this->
visit(Callee))
5862 if (!this->emitSetLocal(
PT_Ptr, *CalleeOffset, E))
5871 if (IsAssignmentOperatorCall) {
5872 assert(Args.size() == 2);
5875 if (!this->emitFlip(Arg2T, Arg1T, E))
5889 assert(HasRVO ==
Func->hasRVO());
5891 bool HasQualifier =
false;
5892 if (
const auto *ME = dyn_cast<MemberExpr>(E->
getCallee()))
5893 HasQualifier = ME->hasQualifier();
5895 bool IsVirtual =
false;
5896 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FuncDecl))
5897 IsVirtual = !Devirtualized && MD->isVirtual();
5902 if (IsVirtual && !HasQualifier) {
5903 uint32_t VarArgSize = 0;
5904 unsigned NumParams =
5905 Func->getNumWrittenParams() +
5907 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5910 if (!this->emitCallVirt(
Func, VarArgSize, E))
5912 }
else if (
Func->isVariadic()) {
5913 uint32_t VarArgSize = 0;
5914 unsigned NumParams =
5915 Func->getNumWrittenParams() +
5917 for (
unsigned I = NumParams, N = E->
getNumArgs(); I != N; ++I)
5919 if (!this->emitCallVar(
Func, VarArgSize, E))
5922 if (!this->emitCall(
Func, 0, E))
5931 uint32_t ArgSize = 0;
5932 for (
unsigned I = 0, N = E->
getNumArgs(); I != N; ++I)
5938 if (!this->emitGetLocal(
PT_MemberPtr, *CalleeOffset, E))
5940 if (!this->emitGetMemberPtrDecl(E))
5943 if (!this->emitGetLocal(
PT_Ptr, *CalleeOffset, E))
5946 if (!this->emitCallPtr(ArgSize, E, E))
5957template <
class Emitter>
5964template <
class Emitter>
5971template <
class Emitter>
5976 return this->emitConstBool(E->
getValue(), E);
5979template <
class Emitter>
5985 uint64_t Val =
Ctx.getASTContext().getTargetNullPointerValue(E->
getType());
5986 return this->emitNullPtr(Val,
nullptr, E);
5989template <
class Emitter>
5997 return this->emitZero(T, E);
6000template <
class Emitter>
6005 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
6006 if (this->LambdaThisCapture.Offset > 0) {
6007 if (this->LambdaThisCapture.IsPtr)
6008 return this->emitGetThisFieldPtr(this->LambdaThisCapture.Offset, E);
6009 return this->emitGetPtrThisField(this->LambdaThisCapture.Offset, E);
6018 return this->emitThis(E);
6033 unsigned StartIndex = 0;
6034 unsigned EndIndex = 0;
6036 for (StartIndex =
InitStack.size() - 1; StartIndex > 0; --StartIndex) {
6038 EndIndex = StartIndex;
6045 for (; StartIndex > 0; --StartIndex) {
6055 if (StartIndex == 0 && EndIndex == 0)
6058 assert(StartIndex < EndIndex);
6061 for (
unsigned I = StartIndex; I != (EndIndex + 1); ++I) {
6073 case Stmt::CompoundStmtClass:
6075 case Stmt::DeclStmtClass:
6077 case Stmt::ReturnStmtClass:
6079 case Stmt::IfStmtClass:
6081 case Stmt::WhileStmtClass:
6083 case Stmt::DoStmtClass:
6085 case Stmt::ForStmtClass:
6087 case Stmt::CXXForRangeStmtClass:
6089 case Stmt::BreakStmtClass:
6091 case Stmt::ContinueStmtClass:
6093 case Stmt::SwitchStmtClass:
6095 case Stmt::CaseStmtClass:
6097 case Stmt::DefaultStmtClass:
6099 case Stmt::AttributedStmtClass:
6101 case Stmt::CXXTryStmtClass:
6103 case Stmt::NullStmtClass:
6106 case Stmt::GCCAsmStmtClass:
6107 case Stmt::MSAsmStmtClass:
6108 case Stmt::GotoStmtClass:
6109 return this->emitInvalid(S);
6110 case Stmt::LabelStmtClass:
6113 if (
const auto *E = dyn_cast<Expr>(S))
6120template <
class Emitter>
6123 for (
const auto *InnerStmt : S->
body())
6126 return Scope.destroyLocals();
6129template <
class Emitter>
6130bool Compiler<Emitter>::maybeEmitDeferredVarInit(
const VarDecl *VD) {
6131 if (
auto *DD = dyn_cast_if_present<DecompositionDecl>(VD)) {
6132 for (
auto *BD : DD->flat_bindings())
6133 if (
auto *KD = BD->getHoldingVar();
6134 KD && !this->visitVarDecl(KD, KD->getInit()))
6148template <
class Emitter>
bool Compiler<Emitter>::refersToUnion(
const Expr *E) {
6150 if (
const auto *ME = dyn_cast<MemberExpr>(E)) {
6151 if (
const auto *FD = dyn_cast<FieldDecl>(ME->getMemberDecl());
6158 if (
const auto *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
6163 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(E);
6164 ICE && (ICE->getCastKind() == CK_NoOp ||
6165 ICE->getCastKind() == CK_DerivedToBase ||
6166 ICE->getCastKind() == CK_UncheckedDerivedToBase)) {
6167 E = ICE->getSubExpr();
6171 if (
const auto *
This = dyn_cast<CXXThisExpr>(E)) {
6172 const auto *ThisRecord =
6173 This->getType()->getPointeeType()->getAsRecordDecl();
6174 if (!ThisRecord->isUnion())
6177 if (
const auto *Ctor =
6178 dyn_cast_if_present<CXXConstructorDecl>(CompilingFunction))
6179 return Ctor->getParent() == ThisRecord;
6188template <
class Emitter>
6190 bool EvaluateConditionDecl) {
6191 for (
const auto *D : DS->
decls()) {
6196 const auto *VD = dyn_cast<VarDecl>(D);
6203 if (EvaluateConditionDecl && !this->maybeEmitDeferredVarInit(VD))
6210template <
class Emitter>
6213 return this->emitUnsupported(RS);
6219 if (!this->
visit(RE))
6225 if (RE->getType()->isVoidType()) {
6226 if (!this->
visit(RE))
6229 if (RE->containsErrors())
6234 if (!this->emitRVOPtr(RE))
6240 return this->emitRetVoid(RS);
6246 return this->emitRetVoid(RS);
6252 auto visitChildStmt = [&](
const Stmt *S) ->
bool {
6259 if (
auto *CondInit = IS->
getInit()) {
6275 return visitChildStmt(IS->
getThen());
6277 return visitChildStmt(Else);
6283 if (!this->emitIsConstantContext(IS))
6286 if (!this->emitIsConstantContext(IS))
6288 if (!this->emitInv(IS))
6302 LabelTy LabelElse = this->getLabel();
6303 LabelTy LabelEnd = this->getLabel();
6304 if (!this->jumpFalse(LabelElse, IS))
6306 if (!visitChildStmt(IS->
getThen()))
6308 if (!this->jump(LabelEnd, IS))
6310 this->emitLabel(LabelElse);
6311 if (!visitChildStmt(Else))
6313 this->emitLabel(LabelEnd);
6315 LabelTy LabelEnd = this->getLabel();
6316 if (!this->jumpFalse(LabelEnd, IS))
6318 if (!visitChildStmt(IS->
getThen()))
6320 this->emitLabel(LabelEnd);
6329template <
class Emitter>
6334 LabelTy CondLabel = this->getLabel();
6335 LabelTy EndLabel = this->getLabel();
6339 this->fallthrough(CondLabel);
6340 this->emitLabel(CondLabel);
6356 if (!this->jumpFalse(EndLabel, S))
6366 if (!this->jump(CondLabel, S))
6368 this->fallthrough(EndLabel);
6369 this->emitLabel(EndLabel);
6378 LabelTy StartLabel = this->getLabel();
6379 LabelTy EndLabel = this->getLabel();
6380 LabelTy CondLabel = this->getLabel();
6384 this->fallthrough(StartLabel);
6385 this->emitLabel(StartLabel);
6391 this->fallthrough(CondLabel);
6392 this->emitLabel(CondLabel);
6399 if (!this->jumpTrue(StartLabel, S))
6402 this->fallthrough(EndLabel);
6403 this->emitLabel(EndLabel);
6407template <
class Emitter>
6415 LabelTy EndLabel = this->getLabel();
6416 LabelTy CondLabel = this->getLabel();
6417 LabelTy IncLabel = this->getLabel();
6424 this->fallthrough(CondLabel);
6425 this->emitLabel(CondLabel);
6437 if (!this->jumpFalse(EndLabel, S))
6446 this->fallthrough(IncLabel);
6447 this->emitLabel(IncLabel);
6453 if (!this->jump(CondLabel, S))
6457 this->emitLabel(EndLabel);
6463template <
class Emitter>
6473 LabelTy EndLabel = this->getLabel();
6474 LabelTy CondLabel = this->getLabel();
6475 LabelTy IncLabel = this->getLabel();
6490 this->fallthrough(CondLabel);
6491 this->emitLabel(CondLabel);
6494 if (!this->jumpFalse(EndLabel, S))
6505 this->fallthrough(IncLabel);
6506 this->emitLabel(IncLabel);
6511 if (!this->jump(CondLabel, S))
6514 this->fallthrough(EndLabel);
6515 this->emitLabel(EndLabel);
6519template <
class Emitter>
6530 if (LI.BreakLabel) {
6531 TargetLabel = *LI.BreakLabel;
6532 BreakScope = LI.BreakOrContinueScope;
6538 if (LI.Name == TargetLoop) {
6539 TargetLabel = *LI.BreakLabel;
6540 BreakScope = LI.BreakOrContinueScope;
6551 C =
C->getParent()) {
6552 if (!
C->destroyLocals())
6556 return this->jump(*TargetLabel, S);
6559template <
class Emitter>
6570 if (LI.ContinueLabel) {
6571 TargetLabel = *LI.ContinueLabel;
6572 ContinueScope = LI.BreakOrContinueScope;
6578 if (LI.Name == TargetLoop) {
6579 TargetLabel = *LI.ContinueLabel;
6580 ContinueScope = LI.BreakOrContinueScope;
6585 assert(TargetLabel);
6588 C =
C->getParent()) {
6589 if (!
C->destroyLocals())
6593 return this->jump(*TargetLabel, S);
6596template <
class Emitter>
6599 if (
Cond->containsErrors())
6605 LabelTy EndLabel = this->getLabel();
6610 if (
const auto *CondInit = S->
getInit())
6621 if (!this->emitSetLocal(CondT, CondVar, S))
6631 if (
const auto *CS = dyn_cast<CaseStmt>(SC)) {
6634 if (CS->caseStmtIsGNURange()) {
6635 LabelTy EndOfRangeCheck = this->getLabel();
6636 const Expr *Low = CS->getLHS();
6637 const Expr *High = CS->getRHS();
6641 if (!this->emitGetLocal(CondT, CondVar, CS))
6643 if (!this->
visit(Low))
6646 if (!this->emitGE(
LT, S))
6648 if (!this->jumpFalse(EndOfRangeCheck, S))
6651 if (!this->emitGetLocal(CondT, CondVar, CS))
6653 if (!this->
visit(High))
6656 if (!this->emitLE(HT, S))
6660 this->emitLabel(EndOfRangeCheck);
6665 if (
Value->isValueDependent())
6670 if (!this->emitGetLocal(CondT, CondVar, CS))
6676 if (!this->emitEQ(ValueT, S))
6681 assert(!DefaultLabel);
6682 DefaultLabel = this->getLabel();
6689 if (!this->jump(*DefaultLabel, S))
6692 if (!this->jump(EndLabel, S))
6700 this->fallthrough(EndLabel);
6701 this->emitLabel(EndLabel);
6706template <
class Emitter>
6713template <
class Emitter>
6720 if (LI.DefaultLabel) {
6721 DefaultLabel = *LI.DefaultLabel;
6726 this->emitLabel(DefaultLabel);
6730template <
class Emitter>
6737 if (IsMSVCConstexprAttr && !this->emitPushMSVCCE(S))
6740 if (this->
Ctx.getLangOpts().CXXAssumptions &&
6741 !this->Ctx.getLangOpts().MSVCCompat) {
6743 auto *AA = dyn_cast<CXXAssumeAttr>(A);
6749 const Expr *Assumption = AA->getAssumption();
6760 if (!this->emitAssume(Assumption))
6769 if (IsMSVCConstexprAttr)
6770 return this->emitPopMSVCCE(S);
6774template <
class Emitter>
6780template <
class Emitter>
6781bool Compiler<Emitter>::emitLambdaStaticInvokerBody(
const CXXMethodDecl *MD) {
6788 assert(ClosureClass->
captures().empty());
6792 "A generic lambda's static-invoker function must be a "
6793 "template specialization");
6797 void *InsertPos =
nullptr;
6798 const FunctionDecl *CorrespondingCallOpSpecialization =
6800 assert(CorrespondingCallOpSpecialization);
6801 LambdaCallOp = CorrespondingCallOpSpecialization;
6805 assert(ClosureClass->
captures().empty());
6806 const Function *
Func = this->getFunction(LambdaCallOp);
6809 assert(
Func->hasThisPointer());
6812 if (
Func->hasRVO()) {
6813 if (!this->emitRVOPtr(MD))
6821 if (!this->emitNullPtr(0,
nullptr, MD))
6826 auto It = this->Params.find(PVD);
6827 assert(It != this->Params.end());
6831 PrimType ParamType = this->classify(PVD->getType()).value_or(
PT_Ptr);
6832 if (!this->emitGetParam(ParamType, It->second.Index, MD))
6836 if (!this->emitCall(
Func, 0, LambdaCallOp))
6841 return this->emitRet(*ReturnType, MD);
6844 return this->emitRetVoid(MD);
6847template <
class Emitter>
6848bool Compiler<Emitter>::checkLiteralType(
const Expr *E) {
6849 if (Ctx.getLangOpts().CPlusPlus23)
6859 const Expr *InitExpr =
Init->getInit();
6861 if (!
Init->isWritten() && !
Init->isInClassMemberInitializer() &&
6865 if (
const auto *CE = dyn_cast<CXXConstructExpr>(InitExpr)) {
6875template <
class Emitter>
6877 assert(!ReturnType);
6880 if (!this->emitStartThisLifetime1(Ctor))
6883 auto emitFieldInitializer = [&](
const Record::Field *F,
unsigned FieldOffset,
6884 const Expr *InitExpr,
6887 if (InitExpr->getType().isNull())
6891 if (
Activate && !this->emitActivateThisField(FieldOffset, InitExpr))
6894 if (!this->visit(InitExpr))
6897 if (F->isBitField())
6898 return this->emitInitThisBitField(*T, FieldOffset, F->bitWidth(),
6900 return this->emitInitThisField(*T, FieldOffset, InitExpr);
6905 if (!this->emitGetPtrThisField(FieldOffset, InitExpr))
6908 if (
Activate && !this->emitActivate(InitExpr))
6911 return this->visitInitializerPop(InitExpr);
6915 const Record *
R = this->getRecord(RD);
6918 bool IsUnion =
R->isUnion();
6928 if (!this->emitThis(Ctor))
6931 if (!this->emitGetParam(
PT_Ptr, 0, Ctor))
6934 return this->emitMemcpy(Ctor) && this->emitPopPtr(Ctor) &&
6935 this->emitRetVoid(Ctor);
6938 unsigned FieldInits = 0;
6940 for (
const auto *
Init : Ctor->
inits()) {
6944 const Expr *InitExpr =
Init->getInit();
6946 const Record::Field *F =
R->getField(
Member);
6950 if (!emitFieldInitializer(F, F->Offset, InitExpr, IsUnion))
6954 const auto *BaseDecl =
Base->getAsCXXRecordDecl();
6957 if (
Init->isBaseVirtual()) {
6958 assert(
R->getVirtualBase(BaseDecl));
6959 if (!this->emitGetPtrThisVirtBase(BaseDecl, InitExpr))
6965 const Record::Base *B =
R->getBase(BaseDecl);
6967 if (!this->emitGetPtrThisBase(B->Offset, InitExpr))
6971 if (IsUnion && !this->emitActivate(InitExpr))
6974 if (!this->visitInitializerPop(InitExpr))
6979 unsigned ChainSize = IFD->getChainingSize();
6980 assert(ChainSize >= 2);
6982 unsigned NestedFieldOffset = 0;
6983 const Record::Field *NestedField =
nullptr;
6984 for (
unsigned I = 0; I != ChainSize; ++I) {
6986 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
6987 assert(FieldRecord);
6989 NestedField = FieldRecord->
getField(FD);
6990 assert(NestedField);
6991 IsUnion = IsUnion || FieldRecord->
isUnion();
6993 NestedFieldOffset += NestedField->Offset;
6996 if (I != ChainSize - 1)
6999 assert(NestedField);
7002 if (!emitFieldInitializer(NestedField, NestedFieldOffset, InitExpr,
7007 unsigned InitFieldOffset = 0;
7008 for (
const NamedDecl *ND : IFD->chain().drop_back()) {
7010 const Record *FieldRecord = this->P.getOrCreateRecord(FD->getParent());
7011 assert(FieldRecord);
7012 NestedField = FieldRecord->
getField(FD);
7013 InitFieldOffset += NestedField->Offset;
7014 assert(NestedField);
7015 if (!this->emitGetPtrThisField(InitFieldOffset, InitExpr))
7017 if (!this->emitFinishInitPop(InitExpr))
7021 InitStack.pop_back_n(ChainSize - 1);
7024 assert(
Init->isDelegatingInitializer());
7025 if (!this->emitThis(InitExpr))
7027 if (!this->visitInitializerPop(
Init->getInit()))
7031 if (!
Scope.destroyLocals())
7035 if (FieldInits !=
R->getNumFields()) {
7036 assert(FieldInits < R->getNumFields());
7038 if (!this->emitStartThisLifetime(Ctor))
7045 if (
const auto *CS = dyn_cast<CompoundStmt>(Body)) {
7046 if (!CS->body_empty() && !this->emitCtorCheck(
SourceInfo{}))
7053 if (!visitStmt(Body))
7060template <
class Emitter>
7063 const Record *
R = this->getRecord(RD);
7068 if (!this->visitStmt(Dtor->
getBody()))
7072 if (!this->emitThis(Dtor))
7075 if (!this->emitCheckDestruction(Dtor))
7079 if (!
R->isUnion()) {
7083 for (
const Record::Field &Field : llvm::reverse(
R->fields())) {
7087 if (!this->emitGetPtrField(Field.Offset,
SourceInfo{}))
7089 if (!this->emitDestructionPop(D,
SourceInfo{}))
7094 for (
const Record::Base &
Base : llvm::reverse(
R->bases())) {
7095 if (
Base.R->hasTrivialDtor())
7099 if (!this->emitRecordDestructionPop(
Base.R, {}))
7103 if (!this->emitMarkDestroyed(Dtor))
7107 return this->emitPopPtr(Dtor) && this->emitRetVoid(Dtor);
7110template <
class Emitter>
7111bool Compiler<Emitter>::compileUnionAssignmentOperator(
7113 if (!this->emitThis(MD))
7116 if (!this->emitGetParam(
PT_Ptr, 0, MD))
7119 return this->emitMemcpy(MD) && this->emitRet(
PT_Ptr, MD);
7122template <
class Emitter>
7132 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(F))
7133 return this->compileConstructor(Ctor);
7134 if (
const auto *Dtor = dyn_cast<CXXDestructorDecl>(F))
7135 return this->compileDestructor(Dtor);
7138 if (
const auto *MD = dyn_cast<CXXMethodDecl>(F)) {
7143 return this->compileUnionAssignmentOperator(MD);
7146 return this->emitLambdaStaticInvokerBody(MD);
7150 if (
const auto *Body = F->
getBody())
7164 return FD->getBitWidthValue();
7167template <
class Emitter>
7183 if (!
Ctx.getLangOpts().CPlusPlus14)
7184 return this->emitInvalid(E);
7186 return this->emitError(E);
7188 if (!this->
visit(SubExpr))
7192 if (!this->emitIncPtr(E))
7199 return DiscardResult ? this->emitIncfPop(getFPOptions(E), E)
7200 : this->emitIncf(getFPOptions(E), E);
7212 if (!
Ctx.getLangOpts().CPlusPlus14)
7213 return this->emitInvalid(E);
7215 return this->emitError(E);
7217 if (!this->
visit(SubExpr))
7221 if (!this->emitDecPtr(E))
7228 return DiscardResult ? this->emitDecfPop(getFPOptions(E), E)
7229 : this->emitDecf(getFPOptions(E), E);
7242 if (!
Ctx.getLangOpts().CPlusPlus14)
7243 return this->emitInvalid(E);
7245 return this->emitError(E);
7247 if (!this->
visit(SubExpr))
7251 if (!this->emitLoadPtr(E))
7253 if (!this->emitConstUint8(1, E))
7255 if (!this->emitAddOffsetUint8(E))
7257 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7263 return this->emitIncfPop(getFPOptions(E), E);
7273 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7274 if (!this->emitLoadFloat(E))
7276 APFloat F(TargetSemantics, 1);
7277 if (!this->emitFloat(F, E))
7280 if (!this->emitAddf(getFPOptions(E), E))
7282 if (!this->emitStoreFloat(E))
7294 return E->
isGLValue() || this->emitLoadPop(*T, E);
7297 if (!
Ctx.getLangOpts().CPlusPlus14)
7298 return this->emitInvalid(E);
7300 return this->emitError(E);
7302 if (!this->
visit(SubExpr))
7306 if (!this->emitLoadPtr(E))
7308 if (!this->emitConstUint8(1, E))
7310 if (!this->emitSubOffsetUint8(E))
7312 return DiscardResult ? this->emitStorePopPtr(E) : this->emitStorePtr(E);
7318 return this->emitDecfPop(getFPOptions(E), E);
7328 const auto &TargetSemantics =
Ctx.getFloatSemantics(E->
getType());
7329 if (!this->emitLoadFloat(E))
7331 APFloat F(TargetSemantics, 1);
7332 if (!this->emitFloat(F, E))
7335 if (!this->emitSubf(getFPOptions(E), E))
7337 if (!this->emitStoreFloat(E))
7349 return E->
isGLValue() || this->emitLoadPop(*T, E);
7353 return this->emitError(E);
7356 return this->
discard(SubExpr);
7361 if (!this->emitInv(E))
7365 return this->emitCast(
PT_Bool, ET, E);
7369 return this->emitError(E);
7371 if (!this->
visit(SubExpr))
7373 return DiscardResult ? this->emitPop(*T, E) : this->emitNeg(*T, E);
7376 return this->emitError(E);
7378 if (!this->
visit(SubExpr))
7394 if (!
Ctx.getLangOpts().CPlusPlus) {
7396 if (
const auto *Deref = dyn_cast<UnaryOperator>(
Sub);
7397 Deref && Deref->getOpcode() == UO_Deref)
7398 return this->
delegate(Deref->getSubExpr());
7404 return this->
discard(SubExpr);
7406 if (!this->
visit(SubExpr))
7413 return this->emitNarrowPtr(E);
7418 return this->emitError(E);
7420 if (!this->
visit(SubExpr))
7422 return DiscardResult ? this->emitPop(*T, E) : this->emitComp(*T, E);
7434 : this->visitZeroInitializer(*T, SubExpr->
getType(), SubExpr);
7439 assert(
false &&
"Unhandled opcode");
7445template <
class Emitter>
7451 return this->
discard(SubExpr);
7454 auto prepareResult = [=]() ->
bool {
7459 return this->emitGetPtrLocal(*LocalIndex, E);
7466 unsigned SubExprOffset = ~0u;
7467 auto createTemp = [=, &SubExprOffset]() ->
bool {
7470 if (!this->
visit(SubExpr))
7472 return this->emitSetLocal(
PT_Ptr, SubExprOffset, E);
7476 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7477 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7479 return this->emitArrayElemPop(ElemT, Index, E);
7484 if (!prepareResult())
7488 for (
unsigned I = 0; I != 2; ++I) {
7489 if (!getElem(SubExprOffset, I))
7491 if (!this->emitNeg(ElemT, E))
7493 if (!this->emitInitElem(ElemT, I, E))
7504 if (!this->
visit(SubExpr))
7506 if (!this->emitComplexBoolCast(SubExpr))
7508 if (!this->emitInv(E))
7511 return this->emitCast(
PT_Bool, ET, E);
7515 return this->emitComplexReal(SubExpr);
7518 if (!this->
visit(SubExpr))
7522 if (!this->emitConstUint8(1, E))
7524 return this->emitArrayElemPtrPopUint8(E);
7535 if (!this->emitArrayElem(ElemT, 1, E))
7537 if (!this->emitNeg(ElemT, E))
7539 if (!this->emitInitElem(ElemT, 1, E))
7547 return this->emitInvalid(E);
7553template <
class Emitter>
7559 return this->
discard(SubExpr);
7562 if (UnaryOp == UO_Extension)
7565 if (UnaryOp != UO_Plus && UnaryOp != UO_Minus && UnaryOp != UO_LNot &&
7566 UnaryOp != UO_Not && UnaryOp != UO_AddrOf)
7567 return this->emitInvalid(E);
7570 if (UnaryOp == UO_Plus || UnaryOp == UO_AddrOf)
7577 if (!this->emitGetPtrLocal(*LocalIndex, E))
7582 unsigned SubExprOffset =
7584 if (!this->
visit(SubExpr))
7586 if (!this->emitSetLocal(
PT_Ptr, SubExprOffset, E))
7591 auto getElem = [=](
unsigned Offset,
unsigned Index) ->
bool {
7592 if (!this->emitGetLocal(
PT_Ptr, Offset, E))
7594 return this->emitArrayElemPop(ElemT, Index, E);
7599 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7600 if (!getElem(SubExprOffset, I))
7602 if (!this->emitNeg(ElemT, E))
7604 if (!this->emitInitElem(ElemT, I, E))
7619 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7620 if (!getElem(SubExprOffset, I))
7623 if (!this->emitPrimCast(ElemT,
PT_Bool,
Ctx.getASTContext().BoolTy, E))
7625 if (!this->emitInv(E))
7627 if (!this->emitPrimCast(
PT_Bool, ElemT, VecTy->getElementType(), E))
7629 if (!this->emitNeg(ElemT, E))
7631 if (ElemT != ResultVecElemT &&
7632 !this->emitPrimCast(ElemT, ResultVecElemT, ResultVecTy, E))
7634 if (!this->emitInitElem(ResultVecElemT, I, E))
7640 for (
unsigned I = 0; I != VecTy->getNumElements(); ++I) {
7641 if (!getElem(SubExprOffset, I))
7644 if (!this->emitInv(E))
7647 if (!this->emitComp(ElemT, E))
7650 if (!this->emitInitElem(ElemT, I, E))
7655 llvm_unreachable(
"Unsupported unary operators should be handled up front");
7660template <
class Emitter>
7665 if (
const auto *ECD = dyn_cast<EnumConstantDecl>(D))
7666 return this->emitConst(ECD->getInitVal(), E);
7667 if (
const auto *FuncDecl = dyn_cast<FunctionDecl>(D)) {
7669 return F && this->emitGetFnPtr(F, E);
7671 if (
const auto *TPOD = dyn_cast<TemplateParamObjectDecl>(D)) {
7676 return this->emitInitGlobal(*T, *Index, E);
7679 if (!this->emitGetPtrGlobal(*Index, E))
7683 return this->emitFinishInit(E);
7698 if (
const auto *PVD = dyn_cast<ParmVarDecl>(D)) {
7699 if (
Ctx.getLangOpts().CPlusPlus && !
Ctx.getLangOpts().CPlusPlus11 &&
7704 if (
auto It = this->Params.find(PVD); It != this->Params.end()) {
7705 if (IsReference || !It->second.IsPtr)
7706 return this->emitGetParam(
classifyPrim(E), It->second.Index, E);
7708 return this->emitGetPtrParam(It->second.Index, E);
7711 if (!
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7717 const unsigned Offset = It->second.Offset;
7720 return this->emitGetRefLocal(Offset, E);
7722 return this->emitGetPtrLocal(Offset, E);
7725 if (
auto GlobalIndex =
P.getGlobal(D)) {
7727 if (!
Ctx.getLangOpts().CPlusPlus11)
7728 return this->emitGetGlobal(
classifyPrim(E), *GlobalIndex, E);
7729 if (!
Ctx.getLangOpts().CPlusPlus23)
7730 return this->emitGetGlobalUnchecked(
classifyPrim(E), *GlobalIndex, E);
7731 return this->emitGetRefGlobal(*GlobalIndex, E);
7734 return this->emitGetPtrGlobal(*GlobalIndex, E);
7738 auto revisit = [&](
const VarDecl *VD,
7739 bool IsConstexprUnknown =
true) ->
bool {
7741 IsConstexprUnknown);
7746 if (!this->emitPopCC(E))
7749 if (VarState.notCreated())
7757 if constexpr (!std::is_same_v<Emitter, EvalEmitter>) {
7759 if (
auto It = this->LambdaCaptures.find(D);
7760 It != this->LambdaCaptures.end()) {
7761 auto [Offset, IsPtr] = It->second;
7764 return this->emitGetThisFieldPtr(Offset, E);
7765 return this->emitGetPtrThisField(Offset, E);
7769 if (
const auto *DRE = dyn_cast<DeclRefExpr>(E);
7770 DRE && DRE->refersToEnclosingVariableOrCapture()) {
7771 if (
const auto *VD = dyn_cast<VarDecl>(D); VD && VD->
isInitCapture())
7775 if (
const auto *BD = dyn_cast<BindingDecl>(D))
7776 return this->
visit(BD->getBinding());
7780 return this->emitDummyPtr(D, E);
7784 const auto *VD = dyn_cast<VarDecl>(D);
7786 return this->emitError(E);
7789 if (!
Ctx.getLangOpts().CPlusPlus) {
7793 return revisit(VD,
false);
7794 return this->emitDummyPtr(D, E);
7798 const auto typeShouldBeVisited = [&](
QualType T) ->
bool {
7799 if (T.isConstant(
Ctx.getASTContext()))
7801 return T->isReferenceType();
7805 typeShouldBeVisited(DeclType)) {
7807 Init && !
Init->isValueDependent()) {
7813 (void)
Init->EvaluateAsInitializer(
V,
Ctx.getASTContext(), VD, Notes,
7827 bool IsConstexprUnknown = !DeclType.
isConstant(
Ctx.getASTContext()) &&
7832 return revisit(VD, IsConstexprUnknown);
7833 }
else if (
Ctx.getLangOpts().CPlusPlus23 && IsReference)
7834 return revisit(VD,
true);
7841 return this->emitDummyPtr(
7845template <
class Emitter>
7851template <
class Emitter>
7861 if (!
C->destroyLocals())
7867template <
class Emitter>
7868unsigned Compiler<Emitter>::collectBaseOffset(
const QualType BaseType,
7871 if (
const auto *R = Ty->getPointeeCXXRecordDecl())
7873 return Ty->getAsCXXRecordDecl();
7875 const CXXRecordDecl *BaseDecl = extractRecordDecl(BaseType);
7876 const CXXRecordDecl *DerivedDecl = extractRecordDecl(DerivedType);
7878 return Ctx.collectBaseOffset(BaseDecl, DerivedDecl);
7882template <
class Emitter>
7889 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7894 return this->emitCastFloatingIntegralAP(Ctx.getBitWidth(ToQT),
7895 getFPOptions(E), E);
7897 return this->emitCastFloatingIntegralAPS(Ctx.getBitWidth(ToQT),
7898 getFPOptions(E), E);
7902 return this->emitCastFloatingIntegral(ToT, getFPOptions(E), E);
7907 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7909 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7913 return FromT != ToT ? this->emitCast(FromT, ToT, E) :
true;
7917 const llvm::fltSemantics *ToSem = &Ctx.getFloatSemantics(ToQT);
7918 return this->emitCastIntegralFloating(FromT, ToSem, getFPOptions(E), E);
7925template <
class Emitter>
7928 assert(FromT != ToT);
7931 return this->emitCastAP(FromT, Ctx.getBitWidth(ToQT), E);
7933 return this->emitCastAPS(FromT, Ctx.getBitWidth(ToQT), E);
7935 return this->emitCast(FromT, ToT, E);
7939template <
class Emitter>
7940bool Compiler<Emitter>::emitComplexReal(
const Expr *SubExpr) {
7944 return this->
discard(SubExpr);
7946 if (!this->visit(SubExpr))
7949 if (!this->emitConstUint8(0, SubExpr))
7951 return this->emitArrayElemPtrPopUint8(SubExpr);
7955 return this->emitArrayElemPop(classifyComplexElementType(SubExpr->
getType()),
7959template <
class Emitter>
7960bool Compiler<Emitter>::emitComplexBoolCast(
const Expr *E) {
7961 assert(!DiscardResult);
7965 if (!this->emitArrayElem(ElemT, 0, E))
7968 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7971 if (!this->emitCast(ElemT,
PT_Bool, E))
7976 LabelTy LabelTrue = this->getLabel();
7977 if (!this->jumpTrue(LabelTrue, E))
7980 if (!this->emitArrayElemPop(ElemT, 1, E))
7983 if (!this->emitCastFloatingIntegral(
PT_Bool, getFPOptions(E), E))
7986 if (!this->emitCast(ElemT,
PT_Bool, E))
7990 LabelTy EndLabel = this->getLabel();
7991 this->jump(EndLabel, E);
7993 this->emitLabel(LabelTrue);
7994 if (!this->emitPopPtr(E))
7996 if (!this->emitConstBool(
true, E))
7999 this->fallthrough(EndLabel);
8000 this->emitLabel(EndLabel);
8005template <
class Emitter>
8006bool Compiler<Emitter>::emitComplexComparison(
const Expr *LHS,
const Expr *RHS,
8017 LHSIsComplex =
true;
8018 ElemT = classifyComplexElementType(LHS->
getType());
8019 LHSOffset = allocateLocalPrimitive(LHS,
PT_Ptr,
true);
8020 if (!this->visit(LHS))
8022 if (!this->emitSetLocal(
PT_Ptr, LHSOffset, E))
8025 LHSIsComplex =
false;
8027 LHSOffset = this->allocateLocalPrimitive(LHS, LHST,
true);
8028 if (!this->visit(LHS))
8030 if (!this->emitSetLocal(LHST, LHSOffset, E))
8037 RHSIsComplex =
true;
8038 ElemT = classifyComplexElementType(RHS->
getType());
8039 RHSOffset = allocateLocalPrimitive(RHS,
PT_Ptr,
true);
8040 if (!this->visit(RHS))
8042 if (!this->emitSetLocal(
PT_Ptr, RHSOffset, E))
8045 RHSIsComplex =
false;
8047 RHSOffset = this->allocateLocalPrimitive(RHS, RHST,
true);
8048 if (!this->visit(RHS))
8050 if (!this->emitSetLocal(RHST, RHSOffset, E))
8054 auto getElem = [&](
unsigned LocalOffset,
unsigned Index,
8055 bool IsComplex) ->
bool {
8057 if (!this->emitGetLocal(
PT_Ptr, LocalOffset, E))
8059 return this->emitArrayElemPop(ElemT, Index, E);
8061 return this->emitGetLocal(ElemT, LocalOffset, E);
8064 for (
unsigned I = 0; I != 2; ++I) {
8066 if (!getElem(LHSOffset, I, LHSIsComplex))
8068 if (!getElem(RHSOffset, I, RHSIsComplex))
8071 if (!this->emitEQ(ElemT, E))
8074 if (!this->emitCastBoolUint8(E))
8079 if (!this->emitAddUint8(E))
8081 if (!this->emitConstUint8(2, E))
8085 if (!this->emitEQUint8(E))
8088 if (!this->emitNEUint8(E))
8095 return this->emitCast(
PT_Bool, ResT, E);
8102template <
class Emitter>
8103bool Compiler<Emitter>::emitRecordDestructionPop(
const Record *R,
8106 assert(!
R->hasTrivialDtor());
8109 const Function *DtorFunc = getFunction(Dtor);
8112 assert(DtorFunc->hasThisPointer());
8113 assert(DtorFunc->getNumParams() == 1);
8114 return this->emitCall(DtorFunc, 0, Loc);
8119template <
class Emitter>
8120bool Compiler<Emitter>::emitDestructionPop(
const Descriptor *Desc,
8132 return this->emitPopPtr(Loc);
8134 for (ssize_t I = N - 1; I >= 1; --I) {
8135 if (!this->emitConstUint64(I, Loc))
8137 if (!this->emitArrayElemPtrUint64(Loc))
8139 if (!this->emitDestructionPop(ElemDesc, Loc))
8143 if (!this->emitConstUint64(0, Loc))
8145 if (!this->emitArrayElemPtrPopUint64(Loc))
8147 return this->emitDestructionPop(ElemDesc, Loc);
8152 return this->emitRecordDestructionPop(Desc->
ElemRecord, Loc);
8157template <
class Emitter>
8158bool Compiler<Emitter>::emitDummyPtr(
const DeclTy &D,
const Expr *E,
bool CU) {
8159 assert(!DiscardResult &&
"Should've been checked before");
8160 unsigned DummyID = P.getOrCreateDummy(D, CU);
8162 if (!this->emitGetPtrGlobal(DummyID, E))
8170 return this->emitDecayPtr(
PT_Ptr, PT, E);
8176template <
class Emitter>
8177bool Compiler<Emitter>::emitFloat(
const APFloat &F,
const Expr *E) {
8179 return this->emitConstFloat(
Floating(F), E);
8181 APInt I = F.bitcastToAPInt();
8182 return this->emitConstFloat(
8183 Floating(
const_cast<uint64_t *
>(I.getRawData()),
8184 llvm::APFloatBase::SemanticsToEnum(F.getSemantics())),
8195template <
class Emitter>
8196bool Compiler<Emitter>::emitBuiltinBitCast(
const CastExpr *E) {
8209 if (!this->emitGetPtrLocal(*LocalIndex, E))
8219 if (!this->visit(SubExpr))
8221 }
else if (
OptPrimType FromT = classify(SubExpr)) {
8222 unsigned TempOffset =
8223 allocateLocalPrimitive(SubExpr, *FromT,
true);
8224 if (!this->visit(SubExpr))
8226 if (!this->emitSetLocal(*FromT, TempOffset, E))
8228 if (!this->emitGetPtrLocal(TempOffset, E))
8235 if (!this->emitBitCast(E))
8237 return DiscardResult ? this->emitPopPtr(E) :
true;
8241 const llvm::fltSemantics *TargetSemantics =
nullptr;
8243 TargetSemantics = &Ctx.getFloatSemantics(ToType);
8249 uint32_t ResultBitWidth = std::max(Ctx.getBitWidth(ToType), 8u);
8251 if (!this->emitBitCastPrim(*ToT, ToTypeIsUChar || ToType->
isStdByteType(),
8252 ResultBitWidth, TargetSemantics,
8257 return this->emitPop(*ToT, E);
8266template <
class Emitter>
8267bool Compiler<Emitter>::emitHLSLAggregateSplat(
PrimType SrcT,
8272 unsigned NumElems = 0;
8275 NumElems = VT->getNumElements();
8276 ElemType = VT->getElementType();
8278 NumElems = MT->getNumElementsFlattened();
8279 ElemType = MT->getElementType();
8282 PrimType ElemT = classifyPrim(ElemType);
8283 for (
unsigned I = 0; I != NumElems; ++I) {
8284 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8286 if (!this->emitPrimCast(SrcT, ElemT, ElemType, E))
8288 if (!this->emitInitElem(ElemT, I, E))
8298 QualType ArrElemType = CAT->getElementType();
8299 unsigned ArrSize = CAT->getZExtSize();
8302 for (
unsigned I = 0; I != ArrSize; ++I) {
8303 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8305 if (!this->emitPrimCast(SrcT, *ElemT, ArrElemType, E))
8307 if (!this->emitInitElem(*ElemT, I, E))
8311 for (
unsigned I = 0; I != ArrSize; ++I) {
8312 if (!this->emitConstUint32(I, E))
8314 if (!this->emitArrayElemPtrUint32(E))
8316 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, ArrElemType, E))
8318 if (!this->emitFinishInitPop(E))
8328 const Record *
R = getRecord(DestType);
8332 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8334 const Record::Base *B =
R->getBase(BS.getType());
8336 if (!this->emitGetPtrBase(B->Offset, E))
8338 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, BS.getType(), E))
8340 if (!this->emitFinishInitPop(E))
8345 for (
const Record::Field &F :
R->fields()) {
8346 if (F.isUnnamedBitField())
8349 QualType FieldType = F.Decl->getType();
8351 if (!this->emitGetLocal(SrcT, SrcOffset, E))
8353 if (!this->emitPrimCast(SrcT, *FieldT, FieldType, E))
8355 if (F.isBitField()) {
8356 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8359 if (!this->emitInitField(*FieldT, F.Offset, E))
8363 if (!this->emitGetPtrField(F.Offset, E))
8365 if (!emitHLSLAggregateSplat(SrcT, SrcOffset, FieldType, E))
8367 if (!this->emitPopPtr(E))
8380template <
class Emitter>
8381unsigned Compiler<Emitter>::countHLSLFlatElements(
QualType Ty) {
8384 return VT->getNumElements();
8386 return MT->getNumElementsFlattened();
8390 return CAT->getZExtSize() * countHLSLFlatElements(CAT->getElementType());
8394 const Record *
R = getRecord(Ty);
8398 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8400 Count += countHLSLFlatElements(BS.getType());
8402 for (
const Record::Field &F :
R->fields()) {
8403 if (F.isUnnamedBitField())
8405 Count += countHLSLFlatElements(F.Decl->getType());
8410 if (canClassify(Ty))
8419template <
class Emitter>
8420bool Compiler<Emitter>::emitHLSLFlattenAggregate(
8421 QualType SrcType,
unsigned SrcOffset,
8426 auto saveToLocal = [&](
PrimType T) ->
bool {
8427 unsigned Offset = allocateLocalPrimitive(E, T,
true);
8428 if (!this->emitSetLocal(T, Offset, E))
8430 Elements.push_back({Offset, T});
8436 unsigned Offset = allocateLocalPrimitive(E,
PT_Ptr,
true);
8437 if (!this->emitSetLocal(
PT_Ptr, Offset, E))
8438 return std::nullopt;
8443 unsigned NumElems = 0;
8446 NumElems = VT->getNumElements();
8447 ElemType = VT->getElementType();
8449 NumElems = MT->getNumElementsFlattened();
8450 ElemType = MT->getElementType();
8453 PrimType ElemT = classifyPrim(ElemType);
8454 for (
unsigned I = 0; I != NumElems && Elements.size() < MaxElements; ++I) {
8455 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8457 if (!this->emitArrayElemPop(ElemT, I, E))
8459 if (!saveToLocal(ElemT))
8469 QualType ArrElemType = CAT->getElementType();
8470 unsigned ArrSize = CAT->getZExtSize();
8473 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8474 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8476 if (!this->emitArrayElemPop(*ElemT, I, E))
8478 if (!saveToLocal(*ElemT))
8482 for (
unsigned I = 0; I != ArrSize && Elements.size() < MaxElements; ++I) {
8483 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8485 if (!this->emitConstUint32(I, E))
8487 if (!this->emitArrayElemPtrPopUint32(E))
8492 if (!emitHLSLFlattenAggregate(ArrElemType, *ElemPtrOffset, Elements,
8503 const Record *
R = getRecord(SrcType);
8507 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8509 if (Elements.size() >= MaxElements)
8511 const Record::Base *B =
R->getBase(BS.getType());
8513 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8515 if (!this->emitGetPtrBasePop(B->Offset,
false, E))
8520 if (!emitHLSLFlattenAggregate(BS.getType(), *BasePtrOffset, Elements,
8526 for (
const Record::Field &F :
R->fields()) {
8527 if (Elements.size() >= MaxElements)
8529 if (F.isUnnamedBitField())
8532 QualType FieldType = F.Decl->getType();
8533 if (!this->emitGetLocal(
PT_Ptr, SrcOffset, E))
8535 if (!this->emitGetPtrFieldPop(F.Offset, E))
8539 if (!this->emitLoadPop(*FieldT, E))
8541 if (!saveToLocal(*FieldT))
8545 if (!FieldPtrOffset)
8547 if (!emitHLSLFlattenAggregate(FieldType, *FieldPtrOffset, Elements,
8563template <
class Emitter>
8564bool Compiler<Emitter>::emitHLSLConstructAggregate(
8570 const auto &Src = Elements[ElemIdx++];
8571 if (!this->emitGetLocal(Src.Type, Src.LocalOffset, E))
8573 return this->emitPrimCast(Src.Type, DestT, DestQT, E);
8577 unsigned NumElems = 0;
8580 NumElems = VT->getNumElements();
8581 ElemType = VT->getElementType();
8583 NumElems = MT->getNumElementsFlattened();
8584 ElemType = MT->getElementType();
8587 PrimType DestElemT = classifyPrim(ElemType);
8588 for (
unsigned I = 0; I != NumElems; ++I) {
8589 if (!loadAndCast(DestElemT, ElemType))
8591 if (!this->emitInitElem(DestElemT, I, E))
8601 QualType ArrElemType = CAT->getElementType();
8602 unsigned ArrSize = CAT->getZExtSize();
8605 for (
unsigned I = 0; I != ArrSize; ++I) {
8606 if (!loadAndCast(*ElemT, ArrElemType))
8608 if (!this->emitInitElem(*ElemT, I, E))
8612 for (
unsigned I = 0; I != ArrSize; ++I) {
8613 if (!this->emitConstUint32(I, E))
8615 if (!this->emitArrayElemPtrUint32(E))
8617 if (!emitHLSLConstructAggregate(ArrElemType, Elements, ElemIdx, E))
8619 if (!this->emitFinishInitPop(E))
8629 const Record *
R = getRecord(DestType);
8633 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(
R->getDecl())) {
8635 const Record::Base *B =
R->getBase(BS.getType());
8637 if (!this->emitGetPtrBase(B->Offset, E))
8639 if (!emitHLSLConstructAggregate(BS.getType(), Elements, ElemIdx, E))
8641 if (!this->emitFinishInitPop(E))
8646 for (
const Record::Field &F :
R->fields()) {
8647 if (F.isUnnamedBitField())
8650 QualType FieldType = F.Decl->getType();
8652 if (!loadAndCast(*FieldT, FieldType))
8654 if (F.isBitField()) {
8655 if (!this->emitInitBitField(*FieldT, F.Offset, F.bitWidth(), E))
8658 if (!this->emitInitField(*FieldT, F.Offset, E))
8662 if (!this->emitGetPtrField(F.Offset, E))
8664 if (!emitHLSLConstructAggregate(FieldType, Elements, ElemIdx, E))
8666 if (!this->emitPopPtr(E))
static void emit(Program &P, llvm::SmallVectorImpl< std::byte > &Code, const T &Val, bool &Success)
Helper to write bytecode and bail out if 32-bit offsets become invalid.
static void emitCleanup(CIRGenFunction &cgf, cir::CleanupScopeOp cleanupScope, EHScopeStack::Cleanup *cleanup, EHScopeStack::Cleanup::Flags flags, Address activeFlag)
static uint32_t getBitWidth(const Expr *E)
#define EMIT_ARITH_OP(OP)
static CharUnits AlignOfType(QualType T, const ASTContext &ASTCtx, UnaryExprOrTypeTrait Kind)
static const Expr * stripDerivedToBaseCasts(const Expr *E)
static const Expr * stripCheckedDerivedToBaseCasts(const Expr *E)
static bool hasTrivialDefaultCtorParent(const FieldDecl *FD)
static bool initNeedsOverridenLoc(const CXXCtorInitializer *Init)
Result
Implement __builtin_bit_cast and related operations.
a trap message and trap category.
llvm::APInt getValue() const
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
const LValueBase getLValueBase() const
APValue & getArrayInitializedElt(unsigned I)
ArrayRef< LValuePathEntry > getLValuePath() const
APValue & getStructField(unsigned i)
const FieldDecl * getUnionField() const
unsigned getStructNumFields() const
bool isMemberPointerToDerivedMember() const
unsigned getArrayInitializedElts() const
unsigned getStructNumBases() const
const ValueDecl * getMemberPointerDecl() const
APValue & getUnionValue()
APValue & getArrayFiller()
ArrayRef< const CXXRecordDecl * > getMemberPointerPath() const
bool isMemberPointer() const
unsigned getArraySize() const
@ None
There is no such object (it's outside its lifetime).
bool isNullPointer() const
APValue & getStructBase(unsigned i)
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
unsigned getPreferredTypeAlign(QualType T) const
Return the "preferred" alignment of the specified type T for the current target, in bits.
const LangOptions & getLangOpts() const
unsigned getOpenMPDefaultSimdAlign(QualType T) const
Get default simd alignment of the specified complete type in bits.
TypeInfoChars getTypeInfoDataSizeInChars(QualType T) const
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
llvm::FixedPointSemantics getFixedPointSemantics(QualType Ty) const
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
const VariableArrayType * getAsVariableArrayType(QualType T) const
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
AbstractConditionalOperator - An abstract base class for ConditionalOperator and BinaryConditionalOpe...
Expr * getCond() const
getCond - Return the expression representing the condition for the ?
Expr * getTrueExpr() const
getTrueExpr - Return the subexpression representing the value of the expression if the condition eval...
Expr * getFalseExpr() const
getFalseExpr - Return the subexpression representing the value of the expression if the condition eva...
AddrLabelExpr - The GNU address of label extension, representing &&label.
Represents the index of the current element of an array being initialized by an ArrayInitLoopExpr.
Represents a loop initializing the elements of an array.
llvm::APInt getArraySize() const
OpaqueValueExpr * getCommonExpr() const
Get the common subexpression shared by all initializations (the source array).
Expr * getSubExpr() const
Get the initializer to use for each array element.
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
Expr * getLHS()
An array access can be written A[4] or 4[A] (both are equivalent).
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
uint64_t getValue() const
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Attr - This represents one attribute.
Represents an attribute applied to a statement.
ArrayRef< const Attr * > getAttrs() const
Represents a C++ declaration that introduces decls from somewhere else.
A builtin binary operation expression such as "x + y" or "x <= y".
static bool isLogicalOp(Opcode Opc)
static bool isComparisonOp(Opcode Opc)
static bool isShiftOp(Opcode Opc)
static bool isCommaOp(Opcode Opc)
static Opcode getOpForCompoundAssignment(Opcode Opc)
static bool isPtrMemOp(Opcode Opc)
predicates to categorize the respective opcodes.
static bool isAssignmentOp(Opcode Opc)
static bool isCompoundAssignmentOp(Opcode Opc)
static bool isBitwiseOp(Opcode Opc)
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
BreakStmt - This represents a break.
Represents a C++2a __builtin_bit_cast(T, v) expression.
Represents a base class of a C++ class.
Represents binding an expression to a temporary.
const Expr * getSubExpr() const
A boolean literal, per ([C++ lex.bool] Boolean literals).
Represents a call to a C++ constructor.
bool isElidable() const
Whether this construction is elidable.
Expr * getArg(unsigned Arg)
Return the specified argument.
bool requiresZeroInitialization() const
Whether this construction first requires zero-initialization before the initializer is called.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will (ultimately) call.
unsigned getNumArgs() const
Return the number of arguments to the constructor call.
Represents a C++ constructor within a class.
bool isCopyOrMoveConstructor(unsigned &TypeQuals) const
Determine whether this is a copy or move constructor.
Represents a C++ base or member initializer.
A default argument (C++ [dcl.fct.default]).
A use of a default initializer in a constructor or in aggregate initialization.
Expr * getExpr()
Get the initialization expression that will be used.
Represents a delete expression for memory deallocation and destructor calls, e.g.
FunctionDecl * getOperatorDelete() const
bool isGlobalDelete() const
Represents a C++ destructor within a class.
A C++ dynamic_cast expression (C++ [expr.dynamic.cast]).
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
DeclStmt * getBeginStmt()
DeclStmt * getLoopVarStmt()
DeclStmt * getRangeStmt()
Represents a call to an inherited base class constructor from an inheriting constructor.
CXXConstructorDecl * getConstructor() const
Get the constructor that this expression will call.
Represents a static or instance method of a struct/union/class.
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
bool isMoveAssignmentOperator() const
Determine whether this is a move assignment operator.
bool isCopyAssignmentOperator() const
Determine whether this is a copy-assignment operator, regardless of whether it was declared implicitl...
bool isLambdaStaticInvoker() const
Determine whether this is a lambda closure type's static member function that is used for the result ...
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
QualType getAllocatedType() const
std::optional< Expr * > getArraySize()
This might return std::nullopt even if isArray() returns true, since there might not be an array size...
Expr * getPlacementArg(unsigned I)
unsigned getNumPlacementArgs() const
FunctionDecl * getOperatorNew() const
Expr * getInitializer()
The initializer of this new-expression.
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
The null pointer literal (C++11 [lex.nullptr])
Represents a list-initialization with parenthesis.
MutableArrayRef< Expr * > getInitExprs()
Represents a C++ struct/union/class.
bool hasTrivialDefaultConstructor() const
Determine whether this class has a trivial default constructor (C++11 [class.ctor]p5).
bool isGenericLambda() const
Determine whether this class describes a generic lambda function object (i.e.
capture_const_range captures() const
CXXMethodDecl * getLambdaCallOperator() const
Retrieve the lambda call operator of the closure type if this is a closure type.
A C++ reinterpret_cast expression (C++ [expr.reinterpret.cast]).
A rewritten comparison expression that was originally written using operator syntax.
Expr * getSemanticForm()
Get an equivalent semantic form for this expression.
An expression "T()" which creates an rvalue of a non-class type T.
Implicit construction of a std::initializer_list<T> object from an array temporary within list-initia...
Represents the this expression in C++.
A C++ throw-expression (C++ [except.throw]).
const Expr * getSubExpr() const
CXXTryStmt - A C++ try block, including all handlers.
CompoundStmt * getTryBlock()
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
bool isTypeOperand() const
QualType getTypeOperand(const ASTContext &Context) const
Retrieves the type operand of this typeid() expression after various required adjustments (removing r...
Expr * getExprOperand() const
bool isPotentiallyEvaluated() const
Determine whether this typeid has a type operand which is potentially evaluated, per C++11 [expr....
A Microsoft C++ __uuidof expression, which gets the _GUID that corresponds to the supplied type or ex...
MSGuidDecl * getGuidDecl() const
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
FunctionDecl * getDirectCallee()
If the callee is a FunctionDecl, return it. Otherwise return null.
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this call.
Expr ** getArgs()
Retrieve the call arguments.
QualType getCallReturnType(const ASTContext &Ctx) const
getCallReturnType - Get the return type of the call expr.
CaseStmt - Represent a case statement.
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
path_iterator path_begin()
CastKind getCastKind() const
llvm::iterator_range< path_iterator > path()
Path through the class hierarchy taken by casts between base and derived classes (see implementation ...
const FieldDecl * getTargetUnionField() const
CharUnits - This is an opaque type for sizes expressed in character units.
bool isZero() const
isZero - Test whether the quantity equals zero.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
unsigned getValue() const
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Expr * getChosenSubExpr() const
getChosenSubExpr - Return the subexpression chosen according to the condition.
Complex values, per C99 6.2.5p11.
QualType getElementType() const
CompoundAssignOperator - For compound assignments (e.g.
QualType getComputationLHSType() const
QualType getComputationResultType() const
CompoundLiteralExpr - [C99 6.5.2.5].
const Expr * getInitializer() const
CompoundStmt - This represents a group of statements like { stmt stmt }.
Represents the specialization of a concept - evaluates to a prvalue of type bool.
bool isSatisfied() const
Whether or not the concept with the given arguments was satisfied when the expression was created.
Represents the canonical version of C arrays with a specified constant size.
uint64_t getZExtSize() const
Return the size zero-extended as a uint64_t.
ConstantExpr - An expression that occurs in a constant context and optionally the result of evaluatin...
APValue getAPValueResult() const
bool hasAPValueResult() const
Represents a concrete matrix type with constant number of rows and columns.
ContinueStmt - This represents a continue.
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
Expr * getSrcExpr() const
getSrcExpr - Return the Expr to be converted.
DeclContext * getParent()
getParent - Returns the containing DeclContext.
A reference to a declared variable, function, enum, etc.
DeclStmt - Adaptor class for mixing declarations with statements and expressions.
Decl - This represents one declaration (or definition), e.g.
bool isInvalidDecl() const
OverloadedOperatorKind getCXXOverloadedOperator() const
If this name is the name of an overloadable operator in C++ (e.g., operator+), retrieve the kind of o...
bool isAnyOperatorNew() const
InitListExpr * getUpdater() const
DoStmt - This represents a 'do/while' stmt.
const Expr * getBase() const
Represents a reference to emded data.
ChildElementIter< false > begin()
Represents an expression – generally a full-expression – that introduces cleanups to be run at the en...
This represents one expression.
const Expr * skipRValueSubobjectAdjustments(SmallVectorImpl< const Expr * > &CommaLHS, SmallVectorImpl< SubobjectAdjustment > &Adjustments) const
Walk outwards from an expression we want to bind a reference to and find the expression whose lifetim...
bool isValueDependent() const
Determines whether the value of this expression depends on.
Expr * IgnoreImplicit() LLVM_READONLY
Skip past any implicit AST nodes which might surround this expression until reaching a fixed point.
bool containsErrors() const
Whether this expression contains subexpressions which had errors.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
bool isLValue() const
isLValue - True if this expression is an "l-value" according to the rules of the current language.
bool HasSideEffects(const ASTContext &Ctx, bool IncludePossibleEffects=true) const
HasSideEffects - This routine returns true for all those expressions which have any effect other than...
bool isTemporaryObject(ASTContext &Ctx, const CXXRecordDecl *TempTy) const
Determine whether the result of this expression is a temporary object of the given class type.
bool refersToBitField() const
Returns true if this expression is a gl-value that potentially refers to a bit-field.
An expression trait intrinsic.
ExtVectorElementExpr - This represents access to specific elements of a vector, and may occur on the ...
void getEncodedElementAccess(SmallVectorImpl< uint32_t > &Elts) const
getEncodedElementAccess - Encode the elements accessed into an llvm aggregate Constant of ConstantInt...
Represents a member of a struct/union/class.
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
llvm::APInt getValue() const
Returns an internal integer representation of the literal.
llvm::APFloat getValue() const
ForStmt - This represents a 'for (init;cond;inc)' stmt.
VarDecl * getConditionVariable() const
Retrieve the variable declared in this "for" statement, if any.
DeclStmt * getConditionVariableDeclStmt()
If this ForStmt has a condition variable, return the faux DeclStmt associated with the creation of th...
const Expr * getSubExpr() const
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
bool isFunctionTemplateSpecialization() const
Determine whether this function is a function template specialization.
FunctionTemplateDecl * getDescribedFunctionTemplate() const
Retrieves the function template that is described by this function declaration.
unsigned getBuiltinID(bool ConsiderWrapperFunctions=false) const
Returns a value indicating whether this function corresponds to a builtin function.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
const TemplateArgumentList * getTemplateSpecializationArgs() const
Retrieve the template arguments used to produce this function template specialization from the primar...
bool isUsableAsGlobalAllocationFunctionInConstantEvaluation(UnsignedOrNone *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions described in i...
bool isDefaulted() const
Whether this function is defaulted.
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Declaration of a template function.
FunctionDecl * findSpecialization(ArrayRef< TemplateArgument > Args, void *&InsertPos)
Return the specialization with the provided arguments if it exists, otherwise return the insertion po...
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
Expr * getResultExpr()
Return the result expression of this controlling expression.
IfStmt - This represents an if/then/else.
bool isNonNegatedConsteval() const
bool isNegatedConsteval() const
DeclStmt * getConditionVariableDeclStmt()
If this IfStmt has a condition variable, return the faux DeclStmt associated with the creation of tha...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "if" statement, if any.
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
const Expr * getSubExpr() const
Represents an implicitly-generated value initialization of an object of a given type.
Represents a field injected from an anonymous union/struct into the parent scope.
Describes an C or C++ initializer list.
Expr * getArrayFiller()
If this initializer list initializes an array with more elements than there are initializers in the l...
ArrayRef< Expr * > inits() const
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
capture_init_iterator capture_init_begin()
Retrieve the first initialization argument for this lambda expression (which initializes the first ca...
CXXRecordDecl * getLambdaClass() const
Retrieve the class that corresponds to the lambda.
bool isCompatibleWith(ClangABI Version) const
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
const Stmt * getNamedLoopOrSwitch() const
If this is a named break/continue, get the loop or switch statement that this targets.
APValue & getAsAPValue() const
Get the value of this MSGuidDecl as an APValue.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
StorageDuration getStorageDuration() const
Retrieve the storage duration for the materialized temporary.
Expr * getSubExpr() const
Retrieve the temporary-generating subexpression whose value will be materialized into a glvalue.
ValueDecl * getExtendingDecl()
Get the declaration which triggered the lifetime-extension of this temporary, if any.
LifetimeExtendedTemporaryDecl * getLifetimeExtendedTemporaryDecl()
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
ValueDecl * getMemberDecl() const
Retrieve the member declaration to which this expression refers.
A pointer to member type per C++ 8.3.3 - Pointers to members.
This represents a decl that may have a name.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
Represents a C++ namespace alias.
ObjCArrayLiteral - used for objective-c array containers; as in: @["Hello", NSApp,...
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
ObjCDictionaryLiteral - AST node to represent objective-c dictionary literals; as in:"name" : NSUserN...
ObjCEncodeExpr, used for @encode in Objective-C.
QualType getEncodedType() const
SourceLocation getAtLoc() const
bool isExpressibleAsConstantInitializer() const
ObjCStringLiteral, used for Objective-C string literals i.e.
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Expr * getIndexExpr(unsigned Idx)
const OffsetOfNode & getComponent(unsigned Idx) const
unsigned getNumComponents() const
Helper class for OffsetOfExpr.
unsigned getArrayExprIndex() const
For an array element node, returns the index into the array of expressions.
@ Array
An index into an array.
Kind getKind() const
Determine what kind of offsetof node this is.
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
Expr * getSourceExpr() const
The source expression of an opaque value expression is the expression which originally generated the ...
Expr * getSelectedExpr() const
ParenExpr - This represents a parenthesized expression, e.g.
const Expr * getSubExpr() const
Represents a parameter to a function.
PointerType - C99 6.7.5.1 - Pointer Declarators.
QualType getPointeeType() const
[C99 6.4.2.2] - A predefined identifier such as func.
StringLiteral * getFunctionName()
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
Expr * getResultExpr()
Return the result-bearing expression, or null if there is none.
ArrayRef< Expr * > semantics()
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
QualType withConst() const
bool isNull() const
Return true if this QualType doesn't point to a type yet.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
bool isConstant(const ASTContext &Ctx) const
bool isConstQualified() const
Determine whether this type is const-qualified.
Represents a struct/union/class.
Frontend produces RecoveryExprs on semantic errors that prevent creating other well-formed expression...
Base for LValueReferenceType and RValueReferenceType.
C++2a [expr.prim.req]: A requires-expression provides a concise way to express requirements on templa...
bool isSatisfied() const
Whether or not the requires clause is satisfied.
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
SourceLocation getLocation() const
std::string ComputeName(ASTContext &Context) const
Scope - A scope is a transient data structure that is used while parsing the program.
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
llvm::APSInt getShuffleMaskIdx(unsigned N) const
unsigned getNumSubExprs() const
getNumSubExprs - Return the size of the SubExprs array.
Expr * getExpr(unsigned Index)
getExpr - Return the Expr at the specified index.
Represents an expression that computes the length of a parameter pack.
unsigned getPackLength() const
Retrieve the length of the parameter pack.
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
APValue EvaluateInContext(const ASTContext &Ctx, const Expr *DefaultExpr) const
Return the result of evaluating this SourceLocExpr in the specified (and possibly null) default argum...
Represents a C++11 static_assert declaration.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
CompoundStmt * getSubStmt()
bool Visit(PTR(Stmt) S, ParamTys... P)
Stmt - This represents one statement.
StmtClass getStmtClass() const
StringLiteral - This represents a string literal expression, e.g.
unsigned getLength() const
static StringLiteral * Create(const ASTContext &Ctx, StringRef Str, StringLiteralKind Kind, bool Pascal, QualType Ty, ArrayRef< SourceLocation > Locs)
This is the "fully general" constructor that allows representation of strings formed from one or more...
uint32_t getCodeUnit(size_t i) const
unsigned getCharByteWidth() const
Represents a reference to a non-type template parameter that has been substituted with a template arg...
Expr * getReplacement() const
const SwitchCase * getNextSwitchCase() const
SwitchStmt - This represents a 'switch' stmt.
VarDecl * getConditionVariable()
Retrieve the variable declared in this "switch" statement, if any.
SwitchCase * getSwitchCaseList()
DeclStmt * getConditionVariableDeclStmt()
If this SwitchStmt has a condition variable, return the faux DeclStmt associated with the creation of...
Represents the declaration of a struct/union/class/enum.
bool isCompleteDefinition() const
Return true if this decl has its body fully specified.
A template argument list.
ArrayRef< TemplateArgument > asArray() const
Produce this as an array ref.
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
bool getBoolValue() const
const APValue & getAPValue() const
bool isStoredAsBoolean() const
The base class of the type hierarchy.
bool isBooleanType() const
bool isLiteralType(const ASTContext &Ctx) const
Return true if this is a literal type (C++11 [basic.types]p10)
bool isIncompleteArrayType() const
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isVoidPointerType() const
bool isConstantSizeType() const
Return true if this is not a variable sized type, according to the rules of C99 6....
bool isFunctionPointerType() const
bool isConstantMatrixType() const
bool isPointerType() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
bool isEnumeralType() const
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isIntegralOrEnumerationType() const
Determine whether this type is an integral or enumeration type.
bool isSpecificBuiltinType(unsigned K) const
Test for a particular builtin type.
bool isDependentType() const
Whether this type is a dependent type, meaning that its definition somehow depends on a template para...
bool isAnyComplexType() const
bool isFixedPointType() const
Return true if this is a fixed point type according to ISO/IEC JTC1 SC22 WG14 N1169.
const Type * getBaseElementTypeUnsafe() const
Get the base element type of this type, potentially discarding type qualifiers.
bool isMemberPointerType() const
bool isAtomicType() const
EnumDecl * castAsEnumDecl() const
bool isStdByteType() const
const ArrayType * getAsArrayTypeUnsafe() const
A variant of getAs<> for array types which silently discards qualifiers from the outermost type.
bool isPointerOrReferenceType() const
bool isFunctionType() const
bool isVectorType() const
bool isRealFloatingType() const
Floating point categories.
const T * getAsCanonical() const
If this type is canonically the specified type, return its canonical type cast to that specified type...
bool isFloatingType() const
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
bool isSizelessVectorType() const
Returns true for all scalable vector types.
Base class for declarations which introduce a typedef-name.
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
QualType getArgumentType() const
QualType getTypeOfArgument() const
Gets the argument type, or the type of the argument expression, whichever is appropriate.
bool isArgumentType() const
UnaryExprOrTypeTrait getKind() const
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Expr * getSubExpr() const
bool canOverflow() const
Returns true if the unary operator can cause an overflow.
Represents C++ using-directive.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
bool isWeak() const
Determine whether this symbol is weakly-imported, or declared with the weak or weak-ref attr.
Represents a variable declaration or definition.
bool isConstexpr() const
Whether this variable is (C++11) constexpr.
bool isInitCapture() const
Whether this variable is the implicit variable for a lambda init-capture.
APValue * evaluateValue() const
Attempt to evaluate the value of the initializer attached to this declaration, and produce notes expl...
bool isStaticDataMember() const
Determines whether this is a static data member.
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
bool hasConstantInitialization() const
Determine whether this variable has constant initialization.
bool isStaticLocal() const
Returns true if a variable with function scope is a static local variable.
const Expr * getInit() const
bool isLocalVarDecl() const
Returns true for local variable declarations other than parameters.
const Expr * getAnyInitializer() const
Get the initializer for this variable, no matter which declaration it is attached to.
Represents a GCC generic vector type.
unsigned getNumElements() const
QualType getElementType() const
WhileStmt - This represents a 'while' stmt.
DeclStmt * getConditionVariableDeclStmt()
If this WhileStmt has a condition variable, return the faux DeclStmt associated with the creation of ...
VarDecl * getConditionVariable()
Retrieve the variable declared in this "while" statement, if any.
A memory block, either on the stack or in the heap.
void invokeDtor()
Invokes the Destructor.
Compilation context for expressions.
llvm::SmallVector< InitLink > InitStack
bool VisitArrayInitIndexExpr(const ArrayInitIndexExpr *E)
bool VisitCXXDeleteExpr(const CXXDeleteExpr *E)
bool VisitOffsetOfExpr(const OffsetOfExpr *E)
bool visitContinueStmt(const ContinueStmt *S)
bool VisitCharacterLiteral(const CharacterLiteral *E)
bool visitArrayElemInit(unsigned ElemIndex, const Expr *Init, OptPrimType InitT)
Pointer to the array(not the element!) must be on the stack when calling this.
UnsignedOrNone allocateLocal(DeclTy &&Decl, QualType Ty=QualType(), ScopeKind=ScopeKind::Block)
Allocates a space storing a local given its type.
bool VisitCXXParenListInitExpr(const CXXParenListInitExpr *E)
bool VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E)
bool visitInitializerPop(const Expr *E)
Similar, but will also pop the pointer.
bool VisitCompoundLiteralExpr(const CompoundLiteralExpr *E)
bool visitBool(const Expr *E)
Visits an expression and converts it to a boolean.
bool VisitCXXDefaultInitExpr(const CXXDefaultInitExpr *E)
PrimType classifyPrim(QualType Ty) const
Classifies a known primitive type.
bool VisitTypeTraitExpr(const TypeTraitExpr *E)
bool VisitLambdaExpr(const LambdaExpr *E)
bool VisitMemberExpr(const MemberExpr *E)
llvm::DenseMap< const OpaqueValueExpr *, unsigned > OpaqueExprs
OpaqueValueExpr to location mapping.
bool VisitBinaryOperator(const BinaryOperator *E)
bool visitAttributedStmt(const AttributedStmt *S)
bool VisitPackIndexingExpr(const PackIndexingExpr *E)
bool visitAPValueInitializer(const APValue &Val, const Expr *E, QualType T)
bool VisitArraySubscriptExpr(const ArraySubscriptExpr *E)
bool VisitCallExpr(const CallExpr *E)
std::optional< uint64_t > ArrayIndex
Current argument index. Needed to emit ArrayInitIndexExpr.
bool VisitPseudoObjectExpr(const PseudoObjectExpr *E)
bool VisitCXXReinterpretCastExpr(const CXXReinterpretCastExpr *E)
const Function * getFunction(const FunctionDecl *FD)
Returns a function for the given FunctionDecl.
bool VisitFixedPointBinOp(const BinaryOperator *E)
bool VisitCastExpr(const CastExpr *E)
bool VisitObjCEncodeExpr(const ObjCEncodeExpr *E)
bool VisitFixedPointUnaryOperator(const UnaryOperator *E)
unsigned allocateLocalPrimitive(DeclTy &&Decl, PrimType Ty, bool IsConst, bool IsVolatile=false, ScopeKind SC=ScopeKind::Block)
Creates a local primitive value.
bool VisitComplexUnaryOperator(const UnaryOperator *E)
llvm::DenseMap< const SwitchCase *, LabelTy > CaseMap
bool VisitBlockExpr(const BlockExpr *E)
bool visitAPValue(const APValue &Val, PrimType ValType, const Expr *E)
Visit an APValue.
bool VisitCXXScalarValueInitExpr(const CXXScalarValueInitExpr *E)
bool VisitLogicalBinOp(const BinaryOperator *E)
bool visitCompoundStmt(const CompoundStmt *S)
Context & Ctx
Current compilation context.
bool visitDeclRef(const ValueDecl *D, const Expr *E)
Visit the given decl as if we have a reference to it.
bool visitBreakStmt(const BreakStmt *S)
bool visitExpr(const Expr *E, bool DestroyToplevelScope) override
bool visitForStmt(const ForStmt *S)
bool VisitDeclRefExpr(const DeclRefExpr *E)
bool VisitOpaqueValueExpr(const OpaqueValueExpr *E)
bool VisitArrayInitLoopExpr(const ArrayInitLoopExpr *E)
bool VisitStmtExpr(const StmtExpr *E)
bool VisitBuiltinBitCastExpr(const BuiltinBitCastExpr *E)
bool VisitFixedPointLiteral(const FixedPointLiteral *E)
const FunctionDecl * CompilingFunction
bool VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *E)
VarCreationState visitVarDecl(const VarDecl *VD, const Expr *Init, bool Toplevel=false)
Creates and initializes a variable from the given decl.
VariableScope< Emitter > * VarScope
Current scope.
bool visitDeclAndReturn(const VarDecl *VD, const Expr *Init, bool ConstantContext) override
Toplevel visitDeclAndReturn().
bool VisitCXXNewExpr(const CXXNewExpr *E)
const ValueDecl * InitializingDecl
bool VisitCompoundAssignOperator(const CompoundAssignOperator *E)
bool visit(const Expr *E) override
Evaluates an expression and places the result on the stack.
bool delegate(const Expr *E)
Just pass evaluation on to E.
bool visitLValueExpr(const Expr *E, bool DestroyToplevelScope) override
bool discard(const Expr *E)
Evaluates an expression for side effects and discards the result.
bool VisitCXXDefaultArgExpr(const CXXDefaultArgExpr *E)
CaseMap CaseLabels
Switch case mapping.
Record * getRecord(QualType Ty)
Returns a record from a record or pointer type.
const RecordType * getRecordTy(QualType Ty)
Returns a record type from a record or pointer type.
bool VisitCXXStdInitializerListExpr(const CXXStdInitializerListExpr *E)
bool visitInitList(ArrayRef< const Expr * > Inits, const Expr *ArrayFiller, const Expr *E)
bool VisitSizeOfPackExpr(const SizeOfPackExpr *E)
bool VisitPredefinedExpr(const PredefinedExpr *E)
bool VisitSourceLocExpr(const SourceLocExpr *E)
bool visitDeclStmt(const DeclStmt *DS, bool EvaluateConditionDecl=false)
bool emitCleanup()
Emits scope cleanup instructions.
bool VisitExtVectorElementExpr(const ExtVectorElementExpr *E)
bool VisitObjCStringLiteral(const ObjCStringLiteral *E)
bool VisitArrayTypeTraitExpr(const ArrayTypeTraitExpr *E)
bool visitInitializer(const Expr *E)
Compiles an initializer.
bool visitDtorCall(const VarDecl *VD, const APValue &Value) override
const Expr * SourceLocDefaultExpr
DefaultInit- or DefaultArgExpr, needed for SourceLocExpr.
bool VisitObjCArrayLiteral(const ObjCArrayLiteral *E)
UnsignedOrNone OptLabelTy
bool VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *E)
bool VisitPointerArithBinOp(const BinaryOperator *E)
Perform addition/subtraction of a pointer and an integer or subtraction of two pointers.
bool visitCallArgs(ArrayRef< const Expr * > Args, const FunctionDecl *FuncDecl, bool Activate, bool IsOperatorCall)
bool VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *E)
bool visitDefaultStmt(const DefaultStmt *S)
bool VisitObjCDictionaryLiteral(const ObjCDictionaryLiteral *E)
typename Emitter::LabelTy LabelTy
VarCreationState visitDecl(const VarDecl *VD)
bool VisitCXXDynamicCastExpr(const CXXDynamicCastExpr *E)
bool visitStmt(const Stmt *S)
bool VisitExpressionTraitExpr(const ExpressionTraitExpr *E)
bool VisitVectorUnaryOperator(const UnaryOperator *E)
bool VisitCXXConstructExpr(const CXXConstructExpr *E)
bool VisitCXXNullPtrLiteralExpr(const CXXNullPtrLiteralExpr *E)
bool VisitObjCBoxedExpr(const ObjCBoxedExpr *E)
bool VisitDesignatedInitUpdateExpr(const DesignatedInitUpdateExpr *E)
bool VisitCXXInheritedCtorInitExpr(const CXXInheritedCtorInitExpr *E)
bool VisitRecoveryExpr(const RecoveryExpr *E)
bool VisitRequiresExpr(const RequiresExpr *E)
bool Initializing
Flag inidicating if we're initializing an already created variable.
bool visitReturnStmt(const ReturnStmt *RS)
bool VisitCXXThrowExpr(const CXXThrowExpr *E)
bool VisitSubstNonTypeTemplateParmExpr(const SubstNonTypeTemplateParmExpr *E)
bool VisitChooseExpr(const ChooseExpr *E)
bool visitFunc(const FunctionDecl *F) override
bool visitCXXForRangeStmt(const CXXForRangeStmt *S)
bool visitCaseStmt(const CaseStmt *S)
bool VisitComplexBinOp(const BinaryOperator *E)
llvm::DenseMap< const ValueDecl *, Scope::Local > Locals
Variable to storage mapping.
bool VisitAbstractConditionalOperator(const AbstractConditionalOperator *E)
bool VisitCXXTypeidExpr(const CXXTypeidExpr *E)
UnsignedOrNone allocateTemporary(const Expr *E)
bool VisitBuiltinCallExpr(const CallExpr *E, unsigned BuiltinID)
bool VisitImplicitValueInitExpr(const ImplicitValueInitExpr *E)
bool VisitCXXRewrittenBinaryOperator(const CXXRewrittenBinaryOperator *E)
OptPrimType ReturnType
Type of the expression returned by the function.
bool VisitUnaryOperator(const UnaryOperator *E)
bool VisitFloatCompoundAssignOperator(const CompoundAssignOperator *E)
OptPrimType classify(const Expr *E) const
llvm::SmallVector< LabelInfo > LabelInfoStack
Stack of label information for loops and switch statements.
bool VisitGenericSelectionExpr(const GenericSelectionExpr *E)
bool visitDoStmt(const DoStmt *S)
bool VisitIntegerLiteral(const IntegerLiteral *E)
bool VisitInitListExpr(const InitListExpr *E)
bool VisitVectorBinOp(const BinaryOperator *E)
bool VisitStringLiteral(const StringLiteral *E)
bool VisitParenExpr(const ParenExpr *E)
bool VisitCXXNoexceptExpr(const CXXNoexceptExpr *E)
bool VisitShuffleVectorExpr(const ShuffleVectorExpr *E)
bool VisitPointerCompoundAssignOperator(const CompoundAssignOperator *E)
bool DiscardResult
Flag indicating if return value is to be discarded.
bool VisitEmbedExpr(const EmbedExpr *E)
bool VisitConvertVectorExpr(const ConvertVectorExpr *E)
bool VisitCXXThisExpr(const CXXThisExpr *E)
bool VisitConstantExpr(const ConstantExpr *E)
bool VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *E)
bool visitSwitchStmt(const SwitchStmt *S)
bool VisitCXXUuidofExpr(const CXXUuidofExpr *E)
bool VisitExprWithCleanups(const ExprWithCleanups *E)
bool visitAsLValue(const Expr *E)
bool visitWhileStmt(const WhileStmt *S)
bool visitIfStmt(const IfStmt *IS)
bool VisitAddrLabelExpr(const AddrLabelExpr *E)
bool canClassify(const Expr *E) const
bool VisitFloatingLiteral(const FloatingLiteral *E)
Program & P
Program to link to.
bool VisitMaterializeTemporaryExpr(const MaterializeTemporaryExpr *E)
bool VariablesAreConstexprUnknown
bool VisitGNUNullExpr(const GNUNullExpr *E)
bool VisitImaginaryLiteral(const ImaginaryLiteral *E)
bool VisitSYCLUniqueStableNameExpr(const SYCLUniqueStableNameExpr *E)
bool visitCXXTryStmt(const CXXTryStmt *S)
static bool isUnevaluatedBuiltin(unsigned ID)
Unevaluated builtins don't get their arguments put on the stack automatically.
static bool shouldBeGloballyIndexed(const ValueDecl *VD)
Returns whether we should create a global variable for the given ValueDecl.
Scope used to handle temporaries in toplevel variable declarations.
DeclScope(Compiler< Emitter > *Ctx, const ValueDecl *VD)
Wrapper around fixed point types.
static FixedPoint zero(llvm::FixedPointSemantics Sem)
If a Floating is constructed from Memory, it DOES NOT OWN THAT MEMORY.
bool hasThisPointer() const
bool hasRVO() const
Checks if the first argument is a RVO pointer.
When generating code for e.g.
LocOverrideScope(Compiler< Emitter > *Ctx, SourceInfo NewValue, bool Enabled=true)
Generic scope for local variables.
bool destroyLocals(const Expr *E=nullptr) override
Explicit destruction of local variables.
LocalScope(Compiler< Emitter > *Ctx, ScopeKind Kind=ScopeKind::Block)
Sets the context for break/continue statements.
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
LoopScope(Compiler< Emitter > *Ctx, const Stmt *Name, LabelTy BreakLabel, LabelTy ContinueLabel)
PrimType value_or(PrimType PT) const
Scope used to handle initialization methods.
OptionScope(Compiler< Emitter > *Ctx, bool NewDiscardResult, bool NewInitializing, bool NewToLValue)
Root constructor, compiling or discarding primitives.
Context to manage declaration lifetimes.
Structure/Class descriptor.
bool isUnion() const
Checks if the record is a union.
const Field * getField(unsigned I) const
const Base * getBase(unsigned I) const
bool hasTrivialDtor() const
Returns true for anonymous unions and records with no destructor or for those with a trivial destruct...
Describes the statement/declaration an opcode was generated from.
StmtExprScope(Compiler< Emitter > *Ctx)
typename Compiler< Emitter >::LabelTy LabelTy
typename Compiler< Emitter >::OptLabelTy OptLabelTy
typename Compiler< Emitter >::LabelInfo LabelInfo
typename Compiler< Emitter >::CaseMap CaseMap
SwitchScope(Compiler< Emitter > *Ctx, const Stmt *Name, CaseMap &&CaseLabels, LabelTy BreakLabel, OptLabelTy DefaultLabel)
Scope chain managing the variable lifetimes.
Compiler< Emitter > * Ctx
Compiler instance.
virtual void addLocal(Scope::Local Local)
VariableScope * getParent() const
bool Sub(InterpState &S, CodePtr OpPC)
bool LT(InterpState &S, CodePtr OpPC)
static llvm::RoundingMode getRoundingMode(FPOptions FPO)
llvm::PointerUnion< const Decl *, const Expr * > DeclTy
constexpr bool isSignedType(PrimType T)
bool Div(InterpState &S, CodePtr OpPC)
1) Pops the RHS from the stack.
static bool Activate(InterpState &S, CodePtr OpPC)
constexpr bool isPtrType(PrimType T)
constexpr size_t align(size_t Size)
Aligns a size to the pointer alignment.
bool InitScope(InterpState &S, CodePtr OpPC, uint32_t I)
constexpr bool isIntegerOrBoolType(PrimType T)
static void discard(InterpStack &Stk, PrimType T)
bool LE(InterpState &S, CodePtr OpPC)
PrimType
Enumeration of the primitive types of the VM.
static std::optional< bool > getBoolValue(const Expr *E)
bool Init(InterpState &S, CodePtr OpPC)
bool Mul(InterpState &S, CodePtr OpPC)
size_t primSize(PrimType Type)
Returns the size of a primitive type in bytes.
bool Inc(InterpState &S, CodePtr OpPC, bool CanOverflow)
1) Pops a pointer from the stack 2) Load the value from the pointer 3) Writes the value increased by ...
bool Add(InterpState &S, CodePtr OpPC)
llvm::BitVector collectNonNullArgs(const FunctionDecl *F, ArrayRef< const Expr * > Args)
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
bool hasSpecificAttr(const Container &container)
@ Success
Annotation was successful.
UnaryExprOrTypeTrait
Names for the "expression or type" traits.
@ SD_Static
Static storage duration.
@ SD_FullExpression
Full-expression storage duration (for temporaries).
@ Result
The result type of a method or function.
OptionalUnsigned< unsigned > UnsignedOrNone
U cast(CodeGen::Address addr)
int const char * function
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()