26#include "llvm/ADT/StringExtras.h"
31#if __has_cpp_attribute(clang::musttail)
32#define MUSTTAIL [[clang::musttail]]
33#elif __has_cpp_attribute(msvc::musttail)
34#define MUSTTAIL [[msvc::musttail]]
35#elif __has_attribute(musttail)
36#define MUSTTAIL __attribute__((musttail))
43#if defined(_MSC_VER) || defined(__powerpc__) || !defined(MUSTTAIL) || \
44 defined(__i386__) || defined(__sparc__)
47#define USE_TAILCALLS 0
49#define USE_TAILCALLS 1
53 llvm::report_fatal_error(
"Interpreter cannot return values");
84 S.
FFDiag(E, diag::note_constexpr_var_init_unknown, 1) << VD;
119 S.
FFDiag(Loc, diag::note_constexpr_access_unknown_variable, 1)
121 S.
Note(D->getLocation(), diag::note_declared_at) << D->getSourceRange();
123 S.
FFDiag(Loc, diag::note_constexpr_function_param_value_unknown, 1) << D;
133 }
else if (
const auto *VD = dyn_cast<VarDecl>(D)) {
134 if (!VD->getAnyInitializer()) {
138 S.
FFDiag(Loc, diag::note_constexpr_var_init_non_constant, 1) << VD;
139 S.
Note(VD->getLocation(), diag::note_declared_at);
157 if (
const auto *VarD = dyn_cast<VarDecl>(VD);
158 VarD && VarD->getType().isConstQualified() &&
159 (VarD->isConstexpr() || !VarD->getType()->isArrayType()) &&
160 !VarD->getAnyInitializer()) {
171 S.
FFDiag(Loc, diag::note_constexpr_ltor_non_const_int, 1) << VD;
177 S.
getLangOpts().CPlusPlus11 ? diag::note_constexpr_ltor_non_constexpr
178 : diag::note_constexpr_ltor_non_integral,
190 const auto *MTE = dyn_cast_if_present<MaterializeTemporaryExpr>(
201 S.
FFDiag(E, diag::note_constexpr_access_static_temporary, 1) << AK;
243 const Expr *
const *Args =
nullptr;
244 unsigned NumArgs = 0;
246 if (
const auto *CE = dyn_cast<CallExpr>(CallSite)) {
247 Args = CE->getArgs();
248 NumArgs = CE->getNumArgs();
249 }
else if (
const auto *CE = dyn_cast<CXXConstructExpr>(CallSite)) {
250 Args = CE->getArgs();
251 NumArgs = CE->getNumArgs();
253 assert(
false &&
"Can't get arguments from that expression type");
255 assert(NumArgs >=
Func->getNumWrittenParams());
256 NumVarArgs = NumArgs - (
Func->getNumWrittenParams() +
258 Func->hasImplicitThisParam()));
259 for (
unsigned I = 0; I != NumVarArgs; ++I) {
260 const Expr *A = Args[NumArgs - 1 - I];
270 if (
Func->hasThisPointer() && !
Func->isThisPointerExplicit())
318 while (!
U.isRoot() && !
U.isActive()) {
323 if (
U.getRecord() &&
U.getRecord()->isAnonymousUnion())
330 assert(
C.getBase() ==
U);
344 if (!
U.getFieldDesc()->isUnion())
354 if (
const auto *CXXRD = dyn_cast<CXXRecordDecl>(R->getDecl());
355 CXXRD && !CXXRD->hasTrivialDefaultConstructor()) {
367 assert(!
C.isActive());
368 const FieldDecl *InactiveField =
C.getField();
369 assert(InactiveField);
372 const Record *R =
U.getRecord();
373 assert(R && R->isUnion() &&
"Not a union");
376 for (
const Record::Field &F : R->fields()) {
377 const Pointer &Field =
U.atField(F.Offset);
378 if (Field.isActive()) {
379 ActiveField = Field.getField();
385 S.
FFDiag(Loc, diag::note_constexpr_access_inactive_union_member)
386 << AK << InactiveField << !ActiveField << ActiveField;
413 S.
FFDiag(E, diag::note_constexpr_unsized_array_indexed);
425 S.
FFDiag(Src, diag::note_constexpr_access_null) << AK;
434 S.
FFDiag(Src, diag::note_constexpr_access_deleted_object) << AK;
436 S.
FFDiag(Src, diag::note_constexpr_access_uninit)
457 if (
const auto *VD = dyn_cast_if_present<VarDecl>(S.
EvaluatingDecl);
463 if (T->isIntegralOrEnumerationType()) {
475 ? diag::note_constexpr_ltor_non_constexpr
476 : diag::note_constexpr_ltor_non_integral,
479 S.
Note(D->getLocation(), diag::note_declared_at);
486 if (T->isPointerOrReferenceType()) {
512 S.
FFDiag(Loc, diag::note_constexpr_null_subobject)
523 S.
FFDiag(Loc, diag::note_constexpr_access_past_end)
534 S.
FFDiag(Loc, diag::note_constexpr_past_end_subobject)
545 S.
FFDiag(Loc, diag::note_constexpr_past_end_subobject)
557 if (Offset < PtrOffset && (PtrOffset - Offset) >= MinOffset)
571 S.
CCEDiag(E, diag::note_constexpr_invalid_downcast)
572 << MostDerivedQT << TargetQT;
578 assert(Ptr.
isLive() &&
"Pointer is not live");
596 S.
FFDiag(Loc, diag::note_constexpr_modify_const_type) << Ty;
602 assert(Ptr.
isLive() &&
"Pointer is not live");
613 S.
FFDiag(Loc, diag::note_constexpr_access_mutable, 1) <<
AK_Read << Field;
614 S.
Note(Field->getLocation(), diag::note_declared_at);
646 Loc = F->getLocation();
650 Loc = VD->getLocation();
655 Loc = E->getExprLoc();
659 diag::note_constexpr_access_volatile_obj, 1)
660 << AK << DiagKind << ND;
661 S.
Note(Loc, diag::note_constexpr_volatile_here) << DiagKind;
687 VD && (VD->isConstexpr() || VD->hasGlobalStorage())) {
690 !(S.
getLangOpts().CPlusPlus23 && VD->getType()->isReferenceType())) {
698 S.
FFDiag(Loc, diag::note_constexpr_access_uninit)
700 S.
Note(VD->getLocation(), diag::note_declared_at);
705 if (VD->getAnyInitializer()) {
707 S.
FFDiag(Loc, diag::note_constexpr_var_init_non_constant, 1) << VD;
708 S.
Note(VD->getLocation(), diag::note_declared_at);
748 S.
Note(VD->getLocation(), diag::note_declared_at);
779 diag::note_constexpr_access_volatile_obj, 1)
793 if (!Desc.IsInitialized)
801 diag::note_constexpr_access_volatile_obj, 1)
817 S.
FFDiag(Src, diag::note_constexpr_access_null) << AK;
869 if (
const auto *CLE =
870 dyn_cast_if_present<CompoundLiteralExpr>(Desc->
asExpr())) {
871 if (
QualType CLETy = CLE->getType();
874 diag::note_invalid_subexpr_in_const_expr)
876 S.
Note(CLE->getExprLoc(), diag::note_declared_at);
920 bool WillBeActivated) {
949 bool IsCtorDtor =
false) {
983 StringRef Name = DiagDecl->
getName();
985 Name ==
"__assert_rtn" || Name ==
"__assert_fail" || Name ==
"_wassert";
988 diag::note_constexpr_assert_failed);
995 diag::note_invalid_subexpr_in_const_expr);
1005 const auto *CD = dyn_cast<CXXConstructorDecl>(DiagDecl);
1006 if (CD && CD->isInheritingConstructor()) {
1007 const auto *Inherited = CD->getInheritedConstructor().getConstructor();
1008 if (!Inherited->isConstexpr())
1009 DiagDecl = CD = Inherited;
1014 if (CD && CD->getParent()->isInvalidDecl())
1020 if (CD && CD->isInheritingConstructor()) {
1023 << CD->getInheritedConstructor().getConstructor()->getParent();
1031 if (!IsDefined && !IsExtern && DiagDecl->
isConstexpr() &&
1041 diag::note_constexpr_invalid_function, 1)
1042 << DiagDecl->
isConstexpr() << (bool)CD << DiagDecl;
1056 S.
CCEDiag(Loc, diag::note_constexpr_virtual_call);
1084 diag::note_constexpr_depth_limit_exceeded)
1098 bool IsImplicit =
false;
1099 if (
const auto *TE = dyn_cast<CXXThisExpr>(E))
1100 IsImplicit = TE->isImplicit();
1101 S.
FFDiag(E, diag::note_constexpr_this) << IsImplicit;
1110 APFloat::opStatus Status,
FPOptions FPO) {
1117 S.
CCEDiag(E, diag::note_constexpr_float_arithmetic)
1127 if ((Status & APFloat::opInexact) &&
1132 S.
FFDiag(E, diag::note_constexpr_dynamic_rounding);
1136 if ((Status != APFloat::opOK) &&
1139 FPO.getAllowFEnvAccess())) {
1141 S.
FFDiag(E, diag::note_constexpr_float_arithmetic_strict);
1145 if ((Status & APFloat::opStatus::opInvalidOp) &&
1161 S.
CCEDiag(E, diag::note_constexpr_new);
1168 const Expr *NewExpr) {
1169 if (AllocForm == DeleteForm)
1175 S.
FFDiag(E, diag::note_constexpr_new_delete_mismatch)
1176 <<
static_cast<int>(DeleteForm) <<
static_cast<int>(AllocForm)
1178 S.
Note(NewExpr->
getExprLoc(), diag::note_constexpr_dynamic_alloc_here)
1186 if (isa_and_nonnull<CXXNewExpr>(Source))
1189 if (
const auto *CE = dyn_cast_if_present<CallExpr>(Source);
1190 CE && CE->getBuiltinCallee() == Builtin::BI__builtin_operator_new)
1193 if (
const auto *MCE = dyn_cast_if_present<CXXMemberCallExpr>(Source);
1194 MCE && MCE->getMethodDecl()->
getIdentifier()->isStr(
"allocate"))
1199 S.
FFDiag(Loc, diag::note_constexpr_delete_not_heap_alloc)
1219 S.
FFDiag(Loc, diag::note_constexpr_var_init_non_constant, 1) << VD;
1220 S.
Note(VD->getLocation(), diag::note_declared_at);
1240 S.
FFDiag(E, diag::note_constexpr_modify_global);
1246 const CallExpr *CE,
unsigned ArgSize) {
1249 unsigned Offset = 0;
1251 for (
const Expr *Arg : Args) {
1252 if (NonNullArgs[Index] && Arg->getType()->isPointerType()) {
1256 S.
CCEDiag(Loc, diag::note_non_null_attribute_failed);
1278 S.
FFDiag(Loc, diag::note_constexpr_double_destroy);
1291 return Call(S, OpPC, DtorFunc, 0);
1314 for (
int I =
static_cast<int>(N) - 1; I >= 0; --I) {
1328 return DD->isVirtual();
1333 bool IsGlobalDelete) {
1339 const Expr *Source =
nullptr;
1340 const Block *BlockToDelete =
nullptr;
1355 BlockToDelete = Ptr.
block();
1359 if (std::optional<DynamicAllocator::Form> AllocForm =
1371 if (!DeleteIsArrayForm && Ptr.
getType() != InitialType &&
1374 diag::note_constexpr_delete_base_nonvirt_dtor)
1375 << InitialType << Ptr.
getType();
1382 S.
FFDiag(Loc, diag::note_constexpr_delete_subobject)
1392 if (!DeleteIsArrayForm && !IsGlobalDelete) {
1397 return DD->isVirtual() ? DD->getOperatorDelete() :
nullptr;
1407 diag::note_constexpr_new_non_replaceable)
1414 assert(BlockToDelete);
1420 if (!Allocator.
deallocate(Source, BlockToDelete, S)) {
1423 S.
FFDiag(Loc, diag::note_constexpr_double_delete);
1440 S.
CCEDiag(Loc, diag::note_constexpr_unscoped_enum_out_of_range)
1441 << llvm::toString(
Value, 10) <<
Min.getSExtValue() <<
Max.getSExtValue()
1445 S.
CCEDiag(Loc, diag::note_constexpr_unscoped_enum_out_of_range)
1446 << llvm::toString(
Value, 10) <<
Min.getZExtValue() <<
Max.getZExtValue()
1477 S.
FFDiag(E, diag::note_constexpr_nonliteral) << E->
getType();
1479 S.
FFDiag(E, diag::note_invalid_subexpr_in_const_expr);
1497 if (std::optional<IntPointer> IntPtr =
1510 diag::note_constexpr_access_unreadable_object)
1528 return getField(S, OpPC, Ptr, Off);
1533 return getField(S, OpPC, Ptr, Off);
1537 uint32_t Off,
bool NullOK) {
1567 return getBase(S, OpPC, Ptr.narrow(), Off,
true);
1571 return getBase(S, OpPC, Ptr.narrow(), Off, NullOK);
1575 const Type *TargetType) {
1599 assert(TargetRecord);
1605 << MostDerivedType <<
QualType(TargetType, 0);
1615 assert(
Func->isConstructor());
1617 if (
Func->getParentDecl()->isInvalidDecl())
1630 <<
Func->getParentDecl();
1640 S.
FFDiag(ND->getLocation(), diag::note_constexpr_destroy_out_of_lifetime)
1641 << ND->getNameAsString();
1644 diag::note_constexpr_destroy_out_of_lifetime)
1666 S.
FFDiag(E, diag::note_constexpr_modify_global);
1689 bool SrcIsVoidPtr) {
1693 if (!Ptr.isBlockPointer())
1700 bool HasValidResult = !Ptr.isZero();
1702 if (HasValidResult) {
1712 S.
CCEDiag(E, diag::note_constexpr_invalid_void_star_cast)
1713 << E->getSubExpr()->getType() << S.
getLangOpts().CPlusPlus26
1714 << Ptr.getType().getCanonicalType() << E->getType()->getPointeeType();
1717 S.
CCEDiag(E, diag::note_constexpr_invalid_cast)
1718 << diag::ConstexprInvalidCastKind::CastFrom <<
"'void *'"
1727 << diag::ConstexprInvalidCastKind::ThisConversionOrReinterpret
1745 uint32_t VarArgSize) {
1746 if (
Func->hasThisPointer()) {
1747 size_t ArgSize =
Func->getArgSize() + VarArgSize;
1757 Func->isLambdaCallOperator())) {
1759 Func->isConstructor() ||
Func->isDestructor()))
1767 if (!
Func->isFullyCompiled())
1783 assert(S.
Current == FrameBefore);
1794 uint32_t VarArgSize) {
1801 auto cleanup = [&]() ->
bool {
1806 if (
Func->hasThisPointer()) {
1807 size_t ArgSize =
Func->getArgSize() + VarArgSize;
1824 Func->isLambdaCallOperator()) {
1825 assert(ThisPtr.
isZero());
1828 Func->isConstructor() ||
Func->isDestructor()))
1831 if (
Func->isCopyOrMoveOperator() ||
Func->isCopyOrMoveConstructor()) {
1838 if (!
Func->isConstructor() && !
Func->isDestructor() &&
1848 if (
Func->isConstructor() ||
Func->isDestructor())
1852 if (!
Func->isFullyCompiled())
1875 if (
Func->isConstructor() ||
Func->isDestructor())
1886 assert(S.
Current == FrameBefore);
1901 S.
FFDiag(E, diag::note_constexpr_polymorphic_unknown_dynamic_type)
1908 DynamicDecl =
DynamicType->getPointeeCXXRecordDecl();
1916 return DynamicDecl !=
nullptr;
1952 for (
const Record::Base &B : R->bases()) {
1955 N.Offset = *N.Offset + B.Offset;
1963 bool IsReferenceCast) {
1967 if (Ptr.isConstexprUnknown()) {
1972 S.
FFDiag(E, diag::note_constexpr_polymorphic_unknown_dynamic_type)
1978 if (!Ptr.isBlockPointer())
1985 LimitedPtr = Ptr.stripBaseCasts().view();
1989 assert(LimitedPtr.
block() == Ptr.block());
1996 if (TargetType->isVoidType()) {
2001 assert(!TargetType.isNull());
2002 assert(!TargetType->isVoidType());
2003 assert(TargetType->isRecordType());
2010 assert(P.getRecord());
2014 auto baseIsPrivate = [&](
PtrView P) ->
bool {
2015 if (P.isRoot() || !P.isBaseClass())
2019 getRecord(P.getBase())->isDerivedFrom(getRecord(P), Paths);
2020 assert(std::distance(Paths.
begin(), Paths.
end()) == 1);
2026 DiagPrivateBase = 0,
2029 DiagPrivateSibling = 3
2032 auto diag = [&](
int DiagKind,
QualType ResultType) ->
bool {
2034 if (!IsReferenceCast) {
2040 diag::note_constexpr_dynamic_cast_to_reference_failed)
2041 << DiagKind << ResultType <<
DynamicType << TargetType;
2050 getRecord(LimitedPtr)
2051 ->isDerivedFrom(TargetType->getAsCXXRecordDecl(), Paths);
2052 if (std::distance(Paths.
begin(), Paths.
end()) == 0 &&
2053 !typesMatch(LimitedPtr.
getType(), TargetType)) {
2054 return diag(DiagNoBase, TargetType);
2059 if (baseIsPrivate(Ptr.view()))
2060 return diag(DiagPrivateBase, Ptr.getType());
2062 std::optional<PtrView>
Result;
2064 for (
PtrView Iter = Ptr.view();;) {
2065 if (typesMatch(TargetType, Iter.getType())) {
2070 Iter = Iter.getBase();
2071 if (Iter.isRoot() || !Iter.isBaseClass())
2078 if (baseIsPrivate(*
Result))
2079 return diag(DiagPrivateBase,
Result->getType());
2086 for (
PtrView Iter = LimitedPtr;;) {
2092 Result = Iter.atField(*R.Offset);
2094 }
else if (R.Ambiguous) {
2100 Iter = Iter.getBase();
2101 if (Iter.isRoot() || !Iter.isBaseClass())
2106 return diag(DiagAmbiguous, TargetType);
2110 if (baseIsPrivate(*
Result))
2111 return diag(DiagPrivateSibling, TargetType);
2117 return diag(DiagNoBase, TargetType);
2121 uint32_t VarArgSize) {
2122 assert(
Func->hasThisPointer());
2123 assert(
Func->isVirtual());
2124 size_t ArgSize =
Func->getArgSize() + VarArgSize;
2136 assert(DynamicDecl);
2149 Overrider = InitialFunction;
2159 S.
Note(Callee->getLocation(), diag::note_declared_at);
2163 if (Overrider != InitialFunction) {
2176 if (
Func->getParentDecl()->isDerivedFrom(ThisFieldDecl)) {
2184 if (!
Call(S, OpPC,
Func, VarArgSize))
2189 if (Overrider != InitialFunction &&
2191 InitialFunction->getReturnType()->isPointerOrReferenceType()) {
2199 OverriderPointeeType))
2216 uint32_t BuiltinID) {
2219 if (BuiltinID == Builtin::BI__builtin_operator_new &&
2283 return CallVirt(S, OpPC, F, VarArgSize);
2285 return Call(S, OpPC, F, VarArgSize);
2292 for (
const Record::Field &Fi : R->fields()) {
2302 for (
unsigned I = 0; I != FieldDesc->getNumElems(); ++I) {
2318 if (!Ptr.isBlockPointer())
2329 if (!Ptr.isBlockPointer())
2340 for (
const Record::Field &Fi : R->fields())
2350 for (
unsigned I = 0; I != FieldDesc->getNumElems(); ++I)
2388 std::optional<uint64_t> ArraySize) {
2391 auto directBaseIsUnion = [](
const Pointer &Ptr) ->
bool {
2395 return R && R->isUnion();
2398 if (Ptr.
inUnion() && directBaseIsUnion(Ptr))
2455 NewExpr->getAllocatedType(),
2456 APInt(64,
static_cast<uint64_t
>(*ArraySize),
false),
nullptr,
2459 AllocType = NewExpr->getAllocatedType();
2462 unsigned StorageSize = 1;
2463 unsigned AllocSize = 1;
2464 if (
const auto *CAT = dyn_cast<ConstantArrayType>(AllocType))
2465 AllocSize = CAT->getZExtSize();
2466 if (
const auto *CAT = dyn_cast<ConstantArrayType>(StorageType))
2467 StorageSize = CAT->getZExtSize();
2469 if (AllocSize > StorageSize ||
2473 diag::note_constexpr_placement_new_wrong_type)
2474 << StorageType << AllocType;
2488 if (
const auto *NewExpr = dyn_cast<CXXNewExpr>(E)) {
2489 const FunctionDecl *OperatorNew = NewExpr->getOperatorNew();
2491 if (NewExpr->getNumPlacementArgs() > 0) {
2502 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation()) {
2504 diag::note_constexpr_new_non_replaceable)
2508 NewExpr->getNumPlacementArgs() == 1 &&
2509 !OperatorNew->isReservedGlobalPlacementOperator()) {
2519 const FunctionDecl *OperatorDelete = DeleteExpr->getOperatorDelete();
2521 ->isUsableAsGlobalAllocationFunctionInConstantEvaluation()) {
2523 diag::note_constexpr_new_non_replaceable)
2537 E->
getExprLoc(), diag::warn_fixedpoint_constant_overflow)
2540 S.
CCEDiag(E, diag::note_constexpr_overflow)
2548 diag::err_shufflevector_minus_one_is_undefined_behavior_constexpr)
2554 const Pointer &Ptr,
unsigned BitWidth) {
2556 S.
CCEDiag(E, diag::note_constexpr_invalid_cast)
2608 bool TargetIsUCharOrByte) {
2610 if (!HasIndeterminateBits)
2614 if (TargetIsUCharOrByte)
2619 S.
FFDiag(E, diag::note_constexpr_bit_cast_indet_dest)
2631 if (!ID.IsInitialized) {
2634 diag::note_constexpr_use_uninit_reference);
2644 const Type *TypeInfoType) {
2652 if (!P.isBlockPointer())
2655 if (P.isConstexprUnknown()) {
2660 S.
FFDiag(E, diag::note_constexpr_polymorphic_unknown_dynamic_type)
2666 CanQualType T = P.getDeclPtr().getType()->getCanonicalTypeUnqualified();
2672 Func && (
Func->isConstructor() ||
Func->isDestructor()) &&
2673 P.block() ==
Frame->getThis().block()) {
2675 Func->getParentDecl());
2687 S.
CCEDiag(E, diag::note_constexpr_typeid_polymorphic)
2688 << E->getExprOperand()->getType()
2689 << E->getExprOperand()->getSourceRange();
2703 StringRef LHSStr(LHSLit->getBytes());
2704 unsigned LHSLength = LHSStr.size();
2705 StringRef RHSStr(RHSLit->getBytes());
2706 unsigned RHSLength = RHSStr.size();
2708 int32_t IndexDiff = RHSOffset - LHSOffset;
2709 if (IndexDiff < 0) {
2710 if (
static_cast<int32_t
>(LHSLength) < -IndexDiff)
2712 LHSStr = LHSStr.drop_front(-IndexDiff);
2714 if (
static_cast<int32_t
>(RHSLength) < IndexDiff)
2716 RHSStr = RHSStr.drop_front(IndexDiff);
2719 unsigned ShorterCharWidth;
2722 if (LHSLength < RHSLength) {
2735 for (
unsigned NullByte : llvm::seq(ShorterCharWidth)) {
2736 if (Shorter.size() + NullByte >= Longer.size())
2738 if (Longer[Shorter.size() + NullByte])
2741 return Shorter == Longer.take_front(Shorter.size());
2748 if (!Val.singleWord()) {
2749 uint64_t *NewMemory =
new (S.
P) uint64_t[Val.numWords()];
2750 Val.take(NewMemory);
2754 if (!Val.singleWord()) {
2755 uint64_t *NewMemory =
new (S.
P) uint64_t[Val.numWords()];
2756 Val.take(NewMemory);
2760 if (!Val.singleWord()) {
2761 uint64_t *NewMemory =
new (S.
P) uint64_t[Val.numWords()];
2762 Val.take(NewMemory);
2768 std::copy_n(Val.path(), PathLength, NewPath);
2769 Val.takePath(NewPath);
2773template <
typename T>
2776 if constexpr (std::is_same_v<T, MemberPointer>) {
2780 std::copy_n(Val.path(), PathLength, NewPath);
2781 Val.takePath(NewPath);
2783 auto &Val = Ptr.
deref<T>();
2784 if (!Val.singleWord()) {
2785 uint64_t *NewMemory =
new (S.
P) uint64_t[Val.numWords()];
2786 Val.take(NewMemory);
2793 for (
const Record::Field &Fi : R->fields()) {
2794 if (Fi.Desc->isPrimitive()) {
2796 copyPrimitiveMemory<T>(S, Ptr.atField(Fi.Offset));
2806 unsigned NumElems = D->getNumElems();
2810 if (D->isPrimitiveArray()) {
2814 assert(NumElems >= 1);
2816 bool AllSingleWord =
true;
2818 if (!EP.
deref<T>().singleWord()) {
2820 AllSingleWord =
false;
2825 for (
unsigned I = 1; I != D->getNumElems(); ++I) {
2830 assert(D->isCompositeArray());
2831 for (
unsigned I = 0; I != D->getNumElems(); ++I) {
2856 S.
CCEDiag(Loc, diag::note_constexpr_invalid_cast)
2857 << diag::ConstexprInvalidCastKind::Reinterpret
2861 S.
CCEDiag(Loc, diag::note_constexpr_invalid_cast)
2862 << diag::ConstexprInvalidCastKind::ThisConversionOrReinterpret
2869 S.
FFDiag(E, diag::note_constexpr_access_volatile_type)
2870 <<
AK_Read << E->getSubExpr()->getType();
2878 S.
CCEDiag(Loc, diag::note_constexpr_invalid_cast)
2879 << diag::ConstexprInvalidCastKind::Dynamic;
2882 llvm_unreachable(
"Unhandled CastKind");
2909 if (MemberPtr.getPathLength() >= 2)
2910 Expected = MemberPtr.getPathEntry(MemberPtr.getPathLength() - 2);
2912 Expected = MemberPtr.getRecordDecl();
2925 unsigned OldPathLength = MemberPtr.getPathLength();
2926 unsigned NewPathLength = OldPathLength - 1;
2927 bool IsDerivedMember = NewPathLength != 0;
2929 std::copy_n(MemberPtr.path(), NewPathLength, NewPath);
2932 NewPath, IsDerivedMember));
2940 bool IsDerivedMember) {
2941 unsigned OldPathLength = MemberPtr.getPathLength();
2942 unsigned NewPathLength = OldPathLength + 1;
2945 std::copy_n(MemberPtr.path(), OldPathLength, NewPath);
2949 NewPath, IsDerivedMember));
2958 if (!Ptr.isDerivedMember() && Ptr.hasPath())
2961 bool IsDerivedMember = Ptr.isDerivedMember() || !Ptr.hasPath();
2970 if (!Ptr.isDerivedMember()) {
2987 if (!MP.isBaseCastPossible())
2998 const auto *FD = dyn_cast_if_present<FunctionDecl>(D);
3002 const auto *
Method = dyn_cast<CXXMethodDecl>(FD);
3009 if (!
Base.getRecord() ||
Base.getRecord()->getDecl() != MethodParent)
3026 unsigned NewPathLength = OldPathLength + 1;
3029 std::copy_n(MemberPtr.path(), OldPathLength, NewPath);
3033 MemberPtr.withPath(NewPathLength, NewPath, IsDerived));
3039 return Op == OP_RetVoid || Op == OP_RetValue || Op == OP_NoRet ||
3040 Op == OP_RetSint8 || Op == OP_RetUint8 || Op == OP_RetSint16 ||
3041 Op == OP_RetUint16 || Op == OP_RetSint32 || Op == OP_RetUint32 ||
3042 Op == OP_RetSint64 || Op == OP_RetUint64 || Op == OP_RetIntAP ||
3043 Op == OP_RetIntAPS || Op == OP_RetBool || Op == OP_RetFixedPoint ||
3044 Op == OP_RetPtr || Op == OP_RetMemberPtr || Op == OP_RetFloat ||
3045 Op == OP_EndSpeculation;
3054#define GET_INTERPFN_DISPATCHERS
3055#include "Opcodes.inc"
3056#undef GET_INTERPFN_DISPATCHERS
3061#define GET_INTERPFN_LIST
3062#include "Opcodes.inc"
3063#undef GET_INTERPFN_LIST
3084 return InterpNext(S, PC);
3111 [[maybe_unused]]
CodePtr PCBefore = RealPC;
3112 size_t StackSizeBefore = S.
Stk.
size();
3118 assert(DepthBefore >= 1);
3122 auto SpeculativeInterp = [&S, &PC]() ->
bool {
3145 if (SpeculativeInterp()) {
3150 assert(S.
Stk.
size() == StackSizeBefore);
3156 assert(S.
Stk.
size() == StackSizeBefore);
3172 assert(*RealPC == *PCBefore);
3182 assert(Offset >= ParamSize);
3183 RealPC += Offset - ParamSize;
Defines the clang::ASTContext interface.
Defines the clang::Expr interface and subclasses for C++ expressions.
static const FunctionDecl * getVirtualOperatorDelete(QualType T)
static bool Jmp(InterpState &S, CodePtr &PC, int32_t Offset)
static bool CheckTemporary(InterpState &S, CodePtr OpPC, const Block *B, AccessKinds AK)
static bool CheckGlobal(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
static PRESERVE_NONE bool RetValue(InterpState &S, CodePtr &Ptr)
static bool Jt(InterpState &S, CodePtr &PC, int32_t Offset)
static void diagnoseNonConstVariable(InterpState &S, CodePtr OpPC, const ValueDecl *VD)
static void noteValueLocation(InterpState &S, const Block *B)
static bool diagnoseUnknownDecl(InterpState &S, CodePtr OpPC, const ValueDecl *D)
static bool Jf(InterpState &S, CodePtr &PC, int32_t Offset)
static void diagnoseMissingInitializer(InterpState &S, CodePtr OpPC, const ValueDecl *VD)
static StringRef getIdentifier(const Token &Tok)
#define TYPE_SWITCH_ALLOC(Expr, B)
#define TYPE_SWITCH(Expr, B)
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
bool hasSameFunctionTypeIgnoringExceptionSpec(QualType T, QualType U) const
Determine whether two function types are the same, ignoring exception specifications in cases where t...
QualType getLValueReferenceType(QualType T, bool SpelledAsLValue=true) const
Return the uniqued reference to the type for an lvalue reference to the specified type.
QualType getConstantArrayType(QualType EltTy, const llvm::APInt &ArySize, const Expr *SizeExpr, ArraySizeModifier ASM, unsigned IndexTypeQuals) const
Return the unique reference to the type for a constant array of the specified element type.
QualType getBaseElementType(const ArrayType *VAT) const
Return the innermost element type of an array type.
bool hasSimilarType(QualType T1, QualType T2) const
Determine if two types are similar, according to the C++ rules.
DiagnosticsEngine & getDiagnostics() const
const TargetInfo & getTargetInfo() const
CanQualType getCanonicalTagType(const TagDecl *TD) const
AccessSpecifier Access
The access along this inheritance path.
BasePaths - Represents the set of paths from a derived class to one of its (direct or indirect) bases...
Represents a C++ destructor within a class.
Represents a static or instance method of a struct/union/class.
Represents a C++ struct/union/class.
CXXRecordDecl * getCanonicalDecl() override
Retrieves the "canonical" declaration of the given declaration.
bool isDerivedFrom(const CXXRecordDecl *Base) const
Determine whether this class is derived from the class Base.
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
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.
A reference to a declared variable, function, enum, etc.
Decl - This represents one declaration (or definition), e.g.
bool isInvalidDecl() const
SourceLocation getLocation() const
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
unsigned getNumNegativeBits() const
Returns the width in bits required to store all the negative enumerators of this enum.
void getValueRange(llvm::APInt &Max, llvm::APInt &Min) const
Calculates the [Min,Max) values the enum can store based on the NumPositiveBits and NumNegativeBits.
This represents one expression.
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
LangOptions::FPExceptionModeKind getExceptionMode() const
RoundingMode getRoundingMode() const
Represents a member of a struct/union/class.
Represents a function declaration or definition.
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
QualType getReturnType() const
bool isTrivial() const
Whether this function is "trivial" in some specialized C++ senses.
StorageClass getStorageClass() const
Returns the storage class as written in the source.
bool isConstexpr() const
Whether this is a (C++11) constexpr function or constexpr constructor.
bool isPureVirtual() const
Whether this virtual function is pure, i.e.
bool isUsableAsGlobalAllocationFunctionInConstantEvaluation(UnsignedOrNone *AlignmentParam=nullptr, bool *IsNothrow=nullptr) const
Determines whether this function is one of the replaceable global allocation functions described in i...
FunctionDecl * getDefinition()
Get the definition for this declaration.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
bool isDefined(const FunctionDecl *&Definition, bool CheckForPendingFriendDefinition=false) const
Returns true if the function has a definition that does not need to be instantiated.
@ FPE_Ignore
Assume that floating-point exceptions are masked.
This represents a decl that may have a name.
IdentifierInfo * getIdentifier() const
Get the identifier that names this declaration, if there is one.
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
bool isConstQualified() const
Determine whether this type is const-qualified.
Represents a struct/union/class.
Encodes a location in the source.
Stmt - This represents one statement.
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
TagDecl * getCanonicalDecl() override
Retrieves the "canonical" declaration of the given declaration.
uint64_t getPointerWidth(LangAS AddrSpace) const
Return the width of pointers on this target, for the specified address space.
The base class of the type hierarchy.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isPointerType() const
CanQualType getCanonicalTypeUnqualified() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
bool isReferenceType() 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 isAnyComplexType() const
bool isPointerOrReferenceType() const
bool isRecordType() const
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a variable declaration or definition.
bool isConstexpr() const
Whether this variable is (C++11) constexpr.
bool isStaticDataMember() const
Determines whether this is a static data member.
A memory block, either on the stack or in the heap.
unsigned getSize() const
Returns the size of the block.
bool isExtern() const
Checks if the block is extern.
const Descriptor * getDescriptor() const
Returns the block's descriptor.
bool isStatic() const
Checks if the block has static storage duration.
bool isTemporary() const
Checks if the block is temporary.
std::byte * rawData()
Returns a pointer to the raw data, including metadata.
bool isInitialized() const
Returns whether the data of this block has been initialized via invoking the Ctor func.
UnsignedOrNone getDeclID() const
Returns the declaration ID.
unsigned getEvalID() const
The Evaluation ID this block was created in.
bool isAccessible() const
Pointer into the code segment.
std::enable_if_t<!std::is_pointer< T >::value, T > read()
Reads data and advances the pointer.
Compilation context for expressions.
unsigned collectBaseOffset(const RecordDecl *BaseDecl, const RecordDecl *DerivedDecl) const
const Function * getOrCreateFunction(const FunctionDecl *FuncDecl)
ASTContext & getASTContext() const
Returns the AST context.
OptPrimType classify(QualType T) const
Classifies a type.
const CXXMethodDecl * getOverridingFunction(const CXXRecordDecl *DynamicDecl, const CXXRecordDecl *StaticDecl, const CXXMethodDecl *InitialFunction) const
Manages dynamic memory allocations done during bytecode interpretation.
bool deallocate(const Expr *Source, const Block *BlockToDelete, InterpState &S)
Deallocate the given source+block combination.
std::optional< Form > getAllocationForm(const Expr *Source) const
Checks whether the allocation done at the given source is an array allocation.
Wrapper around fixed point types.
std::string toDiagnosticString(const ASTContext &Ctx) const
If a Floating is constructed from Memory, it DOES NOT OWN THAT MEMORY.
Base class for stack frames, shared between VM and walker.
Scope & getScope(unsigned Idx)
Returns a specific scope.
bool isDestructor() const
Checks if the function is a destructor.
bool isVirtual() const
Checks if the function is virtual.
bool hasNonNullAttr() const
bool isFullyCompiled() const
Checks if the function is fully done compiling.
bool isConstructor() const
Checks if the function is a constructor.
const FunctionDecl * getDecl() const
Returns the original FunctionDecl.
bool hasBody() const
Checks if the function already has a body attached.
bool isThisPointerExplicit() const
unsigned getWrittenArgSize() const
bool isLambdaStaticInvoker() const
Returns whether this function is a lambda static invoker, which we generate custom byte code for.
bool isValid() const
Checks if the function is valid to call.
If an IntegralAP is constructed from Memory, it DOES NOT OWN THAT MEMORY.
Wrapper around numeric types.
static std::enable_if_t<!std::is_same_v< ValT, IntegralKind >, Integral > from(ValT V, unsigned NumBits=0)
Frame storing local variables.
static void free(InterpFrame *F)
const Expr * getExpr(CodePtr PC) const
InterpFrame * Caller
The frame of the previous function.
SourceInfo getSource(CodePtr PC) const
Map a location to a source.
CodePtr getRetPC() const
Returns the return address of the frame.
Block * getLocalBlock(unsigned Offset) const
CodePtr getPC() const
Returns the PC of the frame's code start.
SourceLocation getLocation(CodePtr PC) const
const Pointer & getThis() const
Returns the 'this' pointer.
unsigned MSVCConstexprAllowed
const Function * getFunction() const
Returns the current function.
SourceRange getRange(CodePtr PC) const
bool isBottomFrame() const
bool isRoot() const
Checks if the frame is a root frame - return should quit the interpreter.
bool hasThisPointer() const
Pointer getLocalPointer(unsigned Offset) const
Returns a pointer to a local variables.
unsigned getDepth() const
void destroy(unsigned Idx)
Invokes the destructors for a scope.
bool isStdFunction() const
static size_t allocSize(const Function *F)
Returns the number of bytes needed to allocate an InterpFrame for the given function.
void clearTo(size_t NewSize)
T pop()
Returns the value from the top of the stack and removes it.
void push(Tys &&...Args)
Constructs a value in place on the top of the stack.
size_t size() const
Returns the size of the stack in bytes.
void discard()
Discards the top value from the stack.
T & peek() const
Returns a reference to the value on the top of the stack.
Context & getContext() const
bool initializingBlock(const Block *B) const
DynamicAllocator & getAllocator()
Context & Ctx
Interpreter Context.
bool noteStep(CodePtr OpPC)
Note that a step has been executed.
const unsigned EvalID
ID identifying this evaluation.
InterpStack & Stk
Temporary stack.
const VarDecl * EvaluatingDecl
Declaration we're initializing/evaluting, if any.
InterpFrame * Current
The current frame.
const CXXRecordDecl ** allocMemberPointerPath(unsigned Length)
llvm::SmallVector< PtrView > InitializingPtrs
List of blocks we're currently running either constructors or destructors for.
T allocAP(unsigned BitWidth)
unsigned SpeculationDepth
StdAllocatorCaller getStdAllocatorCaller(StringRef Name) const
bool inConstantContext() const
Program & P
Reference to the module containing all bytecode.
unsigned getPathLength() const
Return the length of the cast path.
PrimType value_or(PrimType PT) const
A pointer to a memory block, live or dead.
Pointer narrow() const
Restricts the scope of an array element pointer.
UnsignedOrNone getDeclID() const
Returns the declaration ID.
Pointer stripBaseCasts() const
Strip base casts from this Pointer.
bool isVolatile() const
Checks if an object or a subfield is volatile.
bool isInitialized() const
Checks if an object was initialized.
bool isStatic() const
Checks if the storage is static.
bool isDynamic() const
Checks if the storage has been dynamically allocated.
const VarDecl * getRootVarDecl() const
bool isZeroSizeArray() const
Checks if the pointer is pointing to a zero-size array.
Pointer atIndex(uint64_t Idx) const
Offsets a pointer inside an array.
bool isDummy() const
Checks if the pointer points to a dummy value.
Pointer atFieldSub(unsigned Off) const
Subtract the given offset from the current Base and Offset of the pointer.
bool isExtern() const
Checks if the storage is extern.
int64_t getIndex() const
Returns the index into an array.
bool isActive() const
Checks if the object is active.
bool isConst() const
Checks if an object or a subfield is mutable.
Pointer atField(unsigned Off) const
Creates a pointer to a field.
T & deref() const
Dereferences the pointer, if it's live.
bool isMutable() const
Checks if the field is mutable.
bool isConstInMutable() const
unsigned getNumElems() const
Returns the number of elements.
bool isUnknownSizeArray() const
Checks if the structure is an array of unknown size.
bool isIntegralPointer() const
QualType getType() const
Returns the type of the innermost field.
bool isArrayElement() const
Checks if the pointer points to an array.
bool pointsToStringLiteral() const
void initialize() const
Initializes a field.
bool isArrayRoot() const
Whether this array refers to an array, but not to the first element.
bool isLive() const
Checks if the pointer is live.
Pointer getBase() const
Returns a pointer to the object of which this pointer is a field.
uint64_t getByteOffset() const
Returns the byte offset from the start.
bool isTypeidPointer() const
std::string toDiagnosticString(const ASTContext &Ctx) const
Converts the pointer to a string usable in diagnostics.
bool isZero() const
Checks if the pointer is null.
bool isConstexprUnknown() const
const IntPointer & asIntPointer() const
bool isRoot() const
Pointer points directly to a block.
const Descriptor * getDeclDesc() const
Accessor for information about the declaration site.
void activate() const
Activates a field.
static bool pointToSameBlock(const Pointer &A, const Pointer &B)
Checks if both given pointers point to the same block.
APValue toAPValue(const ASTContext &ASTCtx) const
Converts the pointer to an APValue.
bool isOnePastEnd() const
Checks if the index is one past end.
uint64_t getIntegerRepresentation() const
bool isPastEnd() const
Checks if the pointer points past the end of the object.
const FieldDecl * getField() const
Returns the field information.
Pointer expand() const
Expands a pointer to the containing array, undoing narrowing.
bool isElementPastEnd() const
Checks if the pointer is an out-of-bounds element pointer.
void startLifetime() const
Start the lifetime of this pointer.
bool isBlockPointer() const
const FunctionPointer & asFunctionPointer() const
const Block * block() const
bool isFunctionPointer() const
Pointer getDeclPtr() const
const Descriptor * getFieldDesc() const
Accessors for information about the innermost field.
bool canBeInitialized() const
If this pointer has an InlineDescriptor we can use to initialize.
Lifetime getLifetime() const
bool isField() const
Checks if the item is a field in an object.
const Record * getRecord() const
Returns the record descriptor of a class.
UnsignedOrNone getCurrentDecl() const
Returns the current declaration ID.
Structure/Class descriptor.
const RecordDecl * getDecl() const
Returns the underlying declaration.
unsigned getNumVirtualBases() const
llvm::iterator_range< LocalVectorTy::const_reverse_iterator > locals_reverse() const
Describes the statement/declaration an opcode was generated from.
bool checkingForUndefinedBehavior() const
Are we checking an expression for overflow?
OptionalDiagnostic Note(SourceLocation Loc, diag::kind DiagId)
Add a note to a prior diagnostic.
OptionalDiagnostic FFDiag(SourceLocation Loc, diag::kind DiagId=diag::note_invalid_subexpr_in_const_expr, unsigned ExtraNotes=0)
Diagnose that the evaluation could not be folded (FF => FoldFailure)
ASTContext & getASTContext() const
bool noteUndefinedBehavior() const
Note that we hit something that was technically undefined behavior, but that we can evaluate past it ...
OptionalDiagnostic CCEDiag(SourceLocation Loc, diag::kind DiagId=diag::note_invalid_subexpr_in_const_expr, unsigned ExtraNotes=0)
Diagnose that the evaluation does not produce a C++11 core constant expression.
const LangOptions & getLangOpts() const
bool checkingPotentialConstantExpression() const
Are we checking whether the expression is a potential constant expression?
Defines the clang::TargetInfo interface.
bool arePotentiallyOverlappingStringLiterals(const Pointer &LHS, const Pointer &RHS)
bool GetPtrFieldPop(InterpState &S, CodePtr OpPC, uint32_t Off)
static bool CheckCallDepth(InterpState &S, CodePtr OpPC)
const InterpFn InterpFunctions[]
static bool diagnoseCallableDecl(InterpState &S, CodePtr OpPC, const FunctionDecl *DiagDecl)
constexpr bool OpReturns(Opcode Op)
static bool getDynamicDecl(InterpState &S, CodePtr OpPC, Pointer TypePtr, const CXXRecordDecl *&DynamicDecl)
bool CastPointerIntegralAPS(InterpState &S, CodePtr OpPC, uint32_t BitWidth)
static bool CheckVolatile(InterpState &S, CodePtr OpPC, const Pointer &Ptr, AccessKinds AK)
bool CastPointerIntegralAP(InterpState &S, CodePtr OpPC, uint32_t BitWidth)
bool CheckInit(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
Checks if a value can be initialized.
bool GetMemberPtrBase(InterpState &S, CodePtr OpPC)
bool CheckFunctionDecl(InterpState &S, CodePtr OpPC, const FunctionDecl *FD)
Opcode. Check if the function decl can be called at compile time.
static bool appendToMemberPointer(InterpState &S, const MemberPointer &MemberPtr, int32_t BaseOffset, const RecordDecl *BaseDecl, bool IsDerivedMember)
bool EndLifetimePop(InterpState &S, CodePtr OpPC)
Ends the lifetime of the pop'd pointer.
static bool CheckCallable(InterpState &S, CodePtr OpPC, const Function *F)
static bool runRecordDestructor(InterpState &S, CodePtr OpPC, const Pointer &BasePtr, const Descriptor *Desc)
bool GetTypeidPtr(InterpState &S, CodePtr OpPC, const Type *TypeInfoType)
bool LT(InterpState &S, CodePtr OpPC)
bool CheckDowncast(InterpState &S, CodePtr OpPC, const Pointer &Ptr, uint32_t Offset)
Checks if the dowcast using the given offset is possible with the given pointer.
bool CheckNewDeleteForms(InterpState &S, CodePtr OpPC, DynamicAllocator::Form AllocForm, DynamicAllocator::Form DeleteForm, const Descriptor *D, const Expr *NewExpr)
Diagnose mismatched new[]/delete or new/delete[] pairs.
static void setLifeStateRecurse(PtrView Ptr, Lifetime L)
bool CheckGlobalLoad(InterpState &S, CodePtr OpPC, const Block *B)
Checks a direct load of a primitive value from a global or local variable.
bool CheckDeclRef(InterpState &S, CodePtr OpPC, const DeclRefExpr *DR)
We aleady know the given DeclRefExpr is invalid for some reason, now figure out why and print appropr...
bool EndLifetime(InterpState &S, CodePtr OpPC)
Ends the lifetime of the peek'd pointer.
bool GetTypeid(InterpState &S, CodePtr OpPC, const Type *TypePtr, const Type *TypeInfoType)
Typeid support.
static PRESERVE_NONE bool BCP(InterpState &S, CodePtr &RealPC, int32_t Offset, PrimType PT)
This is used to implement speculative execution via __builtin_constant_p when we generate bytecode.
static DynamicCastResult findRecordBase(const ASTContext &Ctx, const Record *R, QualType Needle)
static bool CheckWeak(InterpState &S, CodePtr OpPC, const Block *B)
bool CheckConstant(InterpState &S, CodePtr OpPC, const Descriptor *Desc)
Checks if the Descriptor is of a constexpr or const global variable.
bool CheckPointerToIntegralCast(InterpState &S, CodePtr OpPC, const Pointer &Ptr, unsigned BitWidth)
static bool RunDestructors(InterpState &S, CodePtr OpPC, const Block *B)
bool GetPtrField(InterpState &S, CodePtr OpPC, uint32_t Off)
1) Peeks a Pointer 2) Pushes Pointer.atField(Off) on the stack
bool CheckActive(InterpState &S, CodePtr OpPC, const Pointer &Ptr, AccessKinds AK, bool WillActivate)
static bool CheckNonNullArgs(InterpState &S, CodePtr OpPC, const Function *F, const CallExpr *CE, unsigned ArgSize)
bool StartThisLifetime1(InterpState &S, CodePtr OpPC)
PRESERVE_NONE bool EndSpeculation(InterpState &S, CodePtr &OpPC)
static void finishGlobalRecurse(InterpState &S, const Pointer &Ptr)
bool CheckSubobject(InterpState &S, CodePtr OpPC, const Pointer &Ptr, CheckSubobjectKind CSK)
Checks if Ptr is a one-past-the-end pointer.
bool handleFixedPointOverflow(InterpState &S, CodePtr OpPC, const FixedPoint &FP)
bool handleReference(InterpState &S, CodePtr OpPC, Block *B)
bool CheckBitCast(InterpState &S, CodePtr OpPC, const Type *TargetType, bool SrcIsVoidPtr)
static bool getField(InterpState &S, CodePtr OpPC, const Pointer &Ptr, uint32_t Off)
static void startLifetimeRecurse(PtrView Ptr)
static bool hasVirtualDestructor(QualType T)
bool CheckLoad(InterpState &S, CodePtr OpPC, const Pointer &Ptr, AccessKinds AK)
Checks if a value can be loaded from a block.
static bool getBase(InterpState &S, CodePtr OpPC, const Pointer &Ptr, uint32_t Off, bool NullOK)
bool RVOPtr(InterpState &S, CodePtr OpPC)
constexpr size_t align(size_t Size)
Aligns a size to the pointer alignment.
bool CheckBCPResult(InterpState &S, const Pointer &Ptr)
bool CheckDynamicMemoryAllocation(InterpState &S, CodePtr OpPC)
Checks if dynamic memory allocation is available in the current language mode.
bool CheckLive(InterpState &S, CodePtr OpPC, const Pointer &Ptr, AccessKinds AK)
Checks if a pointer is live and accessible.
bool GetPtrDerivedPop(InterpState &S, CodePtr OpPC, uint32_t Off, bool NullOK, const Type *TargetType)
bool DiagTypeid(InterpState &S, CodePtr OpPC)
bool CheckFinalLoad(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
This is not used by any of the opcodes directly.
bool CopyMemberPtrPath(InterpState &S, CodePtr OpPC, const RecordDecl *Entry, bool IsDerived)
Just append the given Entry to the MemberPointer's path.
static void popArg(InterpState &S, const Expr *Arg)
static bool checkConstructor(InterpState &S, CodePtr OpPC, const Function *Func, const Pointer &ThisPtr)
bool PushIgnoreDiags(InterpState &S, CodePtr OpPC)
void diagnoseEnumValue(InterpState &S, CodePtr OpPC, const EnumDecl *ED, const APSInt &Value)
bool CastMemberPtrDerivedPop(InterpState &S, CodePtr OpPC, int32_t Off, const RecordDecl *BaseDecl)
BaseToDerivedMemberPointer.
bool PopIgnoreDiags(InterpState &S, CodePtr OpPC)
bool isConstexprUnknown(const Block *B)
bool CheckMutable(InterpState &S, CodePtr OpPC, PtrView Ptr)
Checks if a pointer points to a mutable field.
bool InvalidDeclRef(InterpState &S, CodePtr OpPC, const DeclRefExpr *DR, bool InitializerFailed)
bool CheckNull(InterpState &S, CodePtr OpPC, const Pointer &Ptr, CheckSubobjectKind CSK)
Checks if a pointer is null.
bool CheckDeleteSource(InterpState &S, CodePtr OpPC, const Expr *Source, const Pointer &Ptr)
Check the source of the pointer passed to delete/delete[] has actually been heap allocated by us.
bool CheckFloatResult(InterpState &S, CodePtr OpPC, const Floating &Result, APFloat::opStatus Status, FPOptions FPO)
Checks if the result of a floating-point operation is valid in the current context.
PrimType
Enumeration of the primitive types of the VM.
bool InterpretBuiltin(InterpState &S, CodePtr OpPC, const CallExpr *Call, uint32_t BuiltinID)
Interpret a builtin function.
bool CallVar(InterpState &S, CodePtr OpPC, const Function *Func, uint32_t VarArgSize)
constexpr bool needsAlloc()
bool InvalidShuffleVectorIndex(InterpState &S, CodePtr OpPC, uint32_t Index)
bool CheckDummy(InterpState &S, CodePtr OpPC, const Block *B, AccessKinds AK)
Checks if a pointer is a dummy pointer.
static bool diagnoseOutOfLifetimeDestroy(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
bool CheckNewTypeMismatch(InterpState &S, CodePtr OpPC, const Expr *E, std::optional< uint64_t > ArraySize)
Check if the initializer and storage types of a placement-new expression match.
bool GetMemberPtr(InterpState &S, CodePtr OpPC, const ValueDecl *D)
bool CheckRange(InterpState &S, CodePtr OpPC, PtrView Ptr, AccessKinds AK)
Checks if a pointer is in range.
bool CheckLiteralType(InterpState &S, CodePtr OpPC, const Type *T)
bool StartThisLifetime(InterpState &S, CodePtr OpPC)
bool CheckArray(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
Checks if the array is offsetable.
bool GetPtrBase(InterpState &S, CodePtr OpPC, uint32_t Off)
bool GetMemberPtrDecl(InterpState &S, CodePtr OpPC)
static void compileFunction(InterpState &S, const Function *Func)
bool CheckThis(InterpState &S, CodePtr OpPC)
Checks the 'this' pointer.
bool FinishInitGlobal(InterpState &S, CodePtr OpPC)
void cleanupAfterFunctionCall(InterpState &S, CodePtr OpPC, const Function *Func)
bool CheckIntegralAddressCast(InterpState &S, CodePtr OpPC, unsigned BitWidth)
static void copyPrimitiveMemory(InterpState &S, const Pointer &Ptr, PrimType T)
bool CastMemberPtrBasePop(InterpState &S, CodePtr OpPC, int32_t Off, const RecordDecl *BaseDecl)
DerivedToBaseMemberPointer.
bool Destroy(InterpState &S, CodePtr OpPC, uint32_t I)
size_t primSize(PrimType Type)
Returns the size of a primitive type in bytes.
bool Free(InterpState &S, CodePtr OpPC, bool DeleteIsArrayForm, bool IsGlobalDelete)
bool InvalidNewDeleteExpr(InterpState &S, CodePtr OpPC, const Expr *E)
bool CallBI(InterpState &S, CodePtr OpPC, const CallExpr *CE, uint32_t BuiltinID)
bool CheckLocalLoad(InterpState &S, CodePtr OpPC, const Block *B)
bool CheckExtern(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
Checks if the variable has externally defined storage.
static bool CheckLifetime(InterpState &S, CodePtr OpPC, Lifetime LT, const Block *B, AccessKinds AK)
bool CheckStore(InterpState &S, CodePtr OpPC, const Pointer &Ptr, bool WillBeActivated)
Checks if a value can be stored in a block.
bool GetPtrBasePop(InterpState &S, CodePtr OpPC, uint32_t Off, bool NullOK)
bool DiagnoseUninitialized(InterpState &S, CodePtr OpPC, const Pointer &Ptr, AccessKinds AK)
llvm::BitVector collectNonNullArgs(const FunctionDecl *F, ArrayRef< const Expr * > Args)
static bool castBackMemberPointer(InterpState &S, const MemberPointer &MemberPtr, int32_t BaseOffset, const RecordDecl *BaseDecl)
bool CallPtr(InterpState &S, CodePtr OpPC, uint32_t ArgSize, const CallExpr *CE)
bool MarkDestroyed(InterpState &S, CodePtr OpPC)
bool CallVirt(InterpState &S, CodePtr OpPC, const Function *Func, uint32_t VarArgSize)
bool CheckConst(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
Checks if a pointer points to const storage.
bool Interpret(InterpState &S)
Interpreter entry point.
bool(*)(InterpState &, CodePtr &PC) PRESERVE_NONE InterpFn
bool InvalidCast(InterpState &S, CodePtr OpPC, CastKind Kind, bool Fatal)
bool CheckDestructor(InterpState &S, CodePtr OpPC, const Pointer &Ptr)
static bool CheckInvoke(InterpState &S, CodePtr OpPC, const Pointer &Ptr, bool IsCtorDtor=false)
bool DynamicCast(InterpState &S, CodePtr OpPC, const Type *DestTypePtr, bool IsReferenceCast)
The JSON file list parser is used to communicate input to InstallAPI.
CanQual< Type > CanQualType
Represents a canonical, potentially-qualified type.
bool isa(CodeGen::Address addr)
@ Ambiguous
Name lookup results in an ambiguity; use getAmbiguityKind to figure out what kind of ambiguity we hav...
@ Success
Annotation was successful.
CheckSubobjectKind
The order of this enum is important for diagnostics.
@ Result
The result type of a method or function.
AccessKinds
Kinds of access we can perform on an object, for diagnostics.
OptionalUnsigned< unsigned > UnsignedOrNone
U cast(CodeGen::Address addr)
__packed_splat4 __packed_splat2 __packed_splat8 __packed_splat4 int32_t
Describes a memory block created by an allocation site.
unsigned getNumElems() const
Returns the number of elements stored in the block.
bool isPrimitive() const
Checks if the descriptor is of a primitive.
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 ValueDecl * asValueDecl() const
const Decl * asDecl() const
const Descriptor *const ElemDesc
Descriptor of the array element.
unsigned getMetadataSize() const
Returns the size of the metadata.
SourceLocation getLocation() const
QualType getDataType(const ASTContext &Ctx) const
unsigned getElemDataSize() const
Returns the element data size, i.e.
bool isPrimitiveArray() const
Checks if the descriptor is of an array of primitives.
const VarDecl * asVarDecl() const
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.
const Expr * asExpr() const
bool isArray() const
Checks if the descriptor is of an array.
void setOffset(unsigned O)
void merge(DynamicCastResult C)
Descriptor used for global variables.
Inline descriptor embedded in structures and arrays.
std::optional< IntPointer > atOffset(const Context &Ctx, unsigned Offset) const
PtrView atField(unsigned Offset) const
const Record * getRecord() const
const Descriptor * getFieldDesc() const
const FieldDecl * getField() const
PtrView atIndex(unsigned Idx) const
void startLifetime() const
const Block * block() const
bool isOnePastEnd() const
void setLifeState(Lifetime L) const
Lifetime getLifetime() const
bool isZeroSizeArray() const