29 :
Pointer(Pointee, Pointee->getDescriptor()->getMetadataSize(),
30 Pointee->getDescriptor()->getMetadataSize()) {}
33 :
Pointer(Pointee, BaseAndOffset, BaseAndOffset) {}
38 assert((
Base == RootPtrMark ||
Base %
alignof(
void *) == 0) &&
"wrong base");
41 BS = {Pointee,
Base,
nullptr,
nullptr};
42 Pointee->addPointer(
this);
46 : Offset(P.Offset), StorageKind(P.StorageKind) {
47 switch (StorageKind) {
54 BS.Pointee->addPointer(
this);
66 switch (StorageKind) {
73 BS.Pointee->replacePointer(&P,
this);
88 if (
Block *Pointee =
BS.Pointee) {
89 Pointee->removePointer(
this);
105 if (
Block *Pointee =
BS.Pointee) {
106 Pointee->removePointer(
this);
107 BS.Pointee =
nullptr;
112 StorageKind = P.StorageKind;
115 switch (StorageKind) {
123 BS.Pointee->addPointer(
this);
138 if (P.isBlockPointer() && this->block() == P.block()) {
144 if (
Block *Pointee =
BS.Pointee) {
145 Pointee->removePointer(
this);
146 BS.Pointee =
nullptr;
151 StorageKind = P.StorageKind;
154 switch (StorageKind) {
162 BS.Pointee->addPointer(
this);
206 else if (
const auto *E = Desc->
asExpr()) {
215 llvm_unreachable(
"Invalid allocation type");
226 if (FD->getParent()->isInvalidDecl())
229 unsigned FieldIndex = FD->getFieldIndex();
235 VD && VD->
getType()->isReferenceType())
256 const auto *Dcl = Desc->
asDecl();
259 if (
const auto *FD = dyn_cast_if_present<FieldDecl>(Dcl))
288 if (
const auto *BaseOrMember = Desc->
asDecl()) {
289 bool IsVirtual =
false;
290 if (
const auto *FD = dyn_cast<FieldDecl>(BaseOrMember)) {
293 }
else if (
const auto *RD = dyn_cast<CXXRecordDecl>(BaseOrMember)) {
311 llvm_unreachable(
"Invalid field type");
319 std::reverse(Path.begin(), Path.end());
328 switch (StorageKind) {
331 OS <<
"(Block) " << B <<
" {";
334 OS <<
"rootptr(" <<
BS.Base <<
"), ";
336 OS <<
BS.Base <<
", ";
341 OS << Offset <<
", ";
351 OS <<
Int.Value <<
" + " << Offset <<
", " <<
Int.Desc;
355 OS <<
"(Fn) { " <<
Fn.Func <<
" + " << Offset <<
" }";
371 switch (StorageKind) {
373 return Int.Value + Offset;
387 Result += getInlineDesc()->Offset;
453 assert(
BS.Pointee &&
"Cannot check if null pointer was initialized");
462 return getInlineDesc()->IsInitialized;
532 getInlineDesc()->LifeState = L;
539 assert(
BS.Pointee &&
"Cannot initialize null pointer");
557 assert(
BS.Base != 0 &&
"Only composite fields can be initialised");
558 getInlineDesc()->IsInitialized =
true;
586 getInitMap().noteAllInitialized();
625 assert(
BS.Base != 0 &&
"Only composite fields can be activated");
629 if (!getInlineDesc()->InUnion)
634 P.getInlineDesc()->IsActive =
true;
635 if (
const Record *R = P.getRecord(); R && !R->isUnion()) {
636 for (
const Record::Field &F : R->fields()) {
637 Pointer FieldPtr = P.atField(F.Offset);
638 if (!FieldPtr.getInlineDesc()->
IsActive)
647 P.getInlineDesc()->IsActive =
false;
649 if (
const Record *R = P.getRecord()) {
650 for (
const Record::Field &F : R->fields()) {
651 Pointer FieldPtr = P.atField(F.Offset);
652 if (FieldPtr.getInlineDesc()->
IsActive)
673 for (
const Record::Field &F : BR->fields()) {
698 if (A.StorageKind != B.StorageKind)
734 return isa_and_nonnull<StringLiteral>(E);
741 if (
const Expr *E =
BS.Pointee->getDescriptor()->asExpr())
746std::optional<std::pair<Pointer, Pointer>>
757 return std::make_pair(A, B);
760 if (P.isArrayElement())
761 return P.expand().getArray();
779 return std::make_pair(CurA, CurB);
785 llvm_unreachable(
"The loop above should've returned.");
791 assert(!ResultType.
isNull());
794 Composite = [&Composite, &Ctx, &ASTCtx](
QualType Ty,
const Pointer &Ptr,
797 Ty = AT->getValueType();
800 if (Ptr.isDummy() || !Ptr.isLive() || !Ptr.isBlockPointer() ||
807 if (
const auto *RT = Ty->getAsCanonical<RecordType>()) {
808 const auto *
Record = Ptr.getRecord();
809 assert(
Record &&
"Missing record descriptor");
812 if (RT->getDecl()->isUnion()) {
816 const Pointer &FP = Ptr.atField(F.Offset);
823 QualType FieldTy = F.Decl->getType();
824 Ok &= Composite(FieldTy, FP,
Value);
838 for (
unsigned I = 0; I != NF; ++I) {
841 const Pointer &FP = Ptr.atField(FD->Offset);
847 QualType FieldTy = FD->Decl->getType();
848 Ok &= Composite(FieldTy, FP,
Value);
852 for (
unsigned I = 0; I != NB; ++I) {
855 const Pointer &BP = Ptr.atField(BD->Offset);
856 Ok &= Composite(BaseTy, BP, R.getStructBase(I));
859 for (
unsigned I = 0; I != NV; ++I) {
863 const Pointer &VP = Ptr.atField(VD->Offset);
864 Ok &= Composite(VirtBaseTy, VP, R.getStructBase(NB + I));
870 if (Ty->isIncompleteArrayType()) {
875 if (
const auto *AT = Ty->getAsArrayTypeUnsafe()) {
876 const size_t NumElems = Ptr.getNumElems();
877 QualType ElemTy = AT->getElementType();
882 for (
unsigned I = 0; I != NumElems; ++I) {
883 APValue &Slot = R.getArrayInitializedElt(I);
885 TYPE_SWITCH(*ElemT, Slot = Ptr.elem<T>(I).toAPValue(ASTCtx));
887 Ok &= Composite(ElemTy, Ptr.atIndex(I).narrow(), Slot);
894 if (Ty->isAnyComplexType()) {
904 auto V1 = Ptr.elem<T>(0);
905 auto V2 = Ptr.elem<T>(1);
906 R =
APValue(V1.toAPSInt(), V2.toAPSInt());
918 if (
const auto *VT = Ty->getAs<
VectorType>()) {
920 assert(Ptr.getFieldDesc()->isPrimitiveArray());
924 Values.reserve(VT->getNumElements());
925 for (
unsigned I = 0; I != VT->getNumElements(); ++I) {
927 { Values.push_back(Ptr.elem<T>(I).toAPValue(ASTCtx)); });
930 assert(Values.size() == VT->getNumElements());
931 R =
APValue(Values.data(), Values.size());
937 assert(Ptr.getFieldDesc()->isPrimitiveArray());
940 unsigned NumElems = MT->getNumElementsFlattened();
943 Values.reserve(NumElems);
944 for (
unsigned I = 0; I != NumElems; ++I) {
946 { Values.push_back(Ptr.elem<T>(I).toAPValue(ASTCtx)); });
949 R =
APValue(Values.data(), MT->getNumRows(), MT->getNumColumns());
953 llvm_unreachable(
"invalid value to return");
978 if (!Composite(ResultType, *
this,
Result))
984 unsigned Offset)
const {
991 const Record::Field *F =
nullptr;
992 for (
auto &It : R->fields()) {
993 if (It.Offset == Offset) {
1003 return std::nullopt;
1007 uint64_t FieldOffset =
1014 unsigned BaseOffset)
const {
1024 for (
const Record::Base &B : R->bases()) {
1025 if (B.Offset == BaseOffset) {
Defines the clang::Expr interface and subclasses for C++ expressions.
#define INT_TYPE_SWITCH(Expr, B)
#define TYPE_SWITCH(Expr, B)
static uint64_t getFieldOffset(const ASTContext &C, const FieldDecl *FD)
static LValueBase getTypeInfo(TypeInfoLValue LV, QualType TypeInfo)
static LValueBase getDynamicAlloc(DynamicAllocLValue LV, QualType Type)
A non-discriminated union of a base, field, or array index.
static LValuePathEntry ArrayIndex(uint64_t Index)
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
std::string getAsString(const ASTContext &Ctx, QualType Ty) const
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
const ASTRecordLayout & getASTRecordLayout(const RecordDecl *D) const
Get or compute information about the layout of the specified record (struct/union/class) D,...
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
CanQualType getCanonicalTagType(const TagDecl *TD) const
ASTRecordLayout - This class contains layout information for one RecordDecl, which is a struct/union/...
uint64_t getFieldOffset(unsigned FieldNo) const
getFieldOffset - Get the offset of the given field index, in bits.
CharUnits getBaseClassOffset(const CXXRecordDecl *Base) const
getBaseClassOffset - Get the offset, in chars, for the given base class.
CharUnits getVBaseClassOffset(const CXXRecordDecl *VBase) const
getVBaseClassOffset - Get the offset, in chars, for the given base class.
CharUnits - This is an opaque type for sizes expressed in character units.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
static CharUnits Zero()
Zero - Construct a CharUnits quantity of zero.
Represents a concrete matrix type with constant number of rows and columns.
bool isInvalidDecl() const
Symbolic representation of a dynamic allocation.
This represents one expression.
Represents a member of a struct/union/class.
unsigned getFieldIndex() const
Returns the index of this field within its record, as appropriate for passing to ASTRecordLayout::get...
const RecordDecl * getParent() const
Returns the parent of this field declaration, which is the struct in which this field is defined.
Represents a function declaration or definition.
A (possibly-)qualified type.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
RecordDecl * getDefinition() const
Returns the RecordDecl that actually defines this struct/union/class.
Symbolic representation of typeid(T) for some type T.
RecordDecl * getAsRecordDecl() const
Retrieves the RecordDecl this type refers to.
bool isFunctionType() const
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a GCC generic vector type.
unsigned getSize() const
Returns the size of the block.
const Descriptor * getDescriptor() const
Returns the block's descriptor.
Holds all information required to evaluate constexpr code in a module.
ASTContext & getASTContext() const
Returns the AST context.
OptPrimType classify(QualType T) const
Classifies a type.
bool canClassify(QualType T) const
If a Floating is constructed from Memory, it DOES NOT OWN THAT MEMORY.
APFloat getAPFloat() const
const BlockExpr * getExpr() const
const FunctionDecl * getDecl() const
Returns the original FunctionDecl.
static bool hasSameBase(const Pointer &A, const Pointer &B)
Checks if two pointers are comparable.
void deactivate() const
Deactivates an entire strurcutre.
bool isInitialized() const
Checks if an object was initialized.
bool pointsToLabel() const
Whether this points to a block created for an AddrLabelExpr.
bool isStatic() const
Checks if the storage is static.
bool isDynamic() const
Checks if the storage has been dynamically allocated.
bool isZeroSizeArray() const
Checks if the pointer is pointing to a zero-size array.
bool isElementInitialized(unsigned Index) const
Like isInitialized(), but for primitive arrays.
bool isDummy() const
Checks if the pointer points to a dummy value.
void print(llvm::raw_ostream &OS) const
Prints the pointer.
int64_t getIndex() const
Returns the index into an array.
bool isActive() const
Checks if the object is active.
void startLifetime() const
Start the lifetime of this pointer.
bool canDeref(PrimType T) const
Checks whether the pointer can be dereferenced to the given PrimType.
Pointer atField(unsigned Off) const
Creates a pointer to a field.
T & deref() const
Dereferences the pointer, if it's live.
unsigned getNumElems() const
Returns the number of elements.
Pointer getArray() const
Returns the parent array.
bool isUnknownSizeArray() const
Checks if the structure is an array of unknown size.
void activate() const
Activats a field.
static std::optional< std::pair< Pointer, Pointer > > computeSplitPoint(const Pointer &A, const Pointer &B)
const TypeidPointer & asTypeidPointer() const
bool isIntegralPointer() const
QualType getType() const
Returns the type of the innermost field.
bool isArrayElement() const
Checks if the pointer points to an array.
void initializeAllElements() const
Initialize all elements of a primitive array at once.
bool pointsToStringLiteral() const
bool isArrayRoot() const
Whether this array refers to an array, but not to the first element.
size_t computeOffsetForComparison(const ASTContext &ASTCtx) const
Compute an integer that can be used to compare this pointer to another one.
bool isLive() const
Checks if the pointer is live.
bool inArray() const
Checks if the innermost field is an array.
bool isElementAlive(unsigned Index) const
bool pointsToLiteral() const
Whether this points to a block that's been created for a "literal lvalue", i.e.
bool allElementsAlive() const
Pointer getBase() const
Returns a pointer to the object of which this pointer is a field.
void setLifeState(Lifetime L) const
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.
Pointer & operator=(const Pointer &P)
const IntPointer & asIntPointer() const
bool isRoot() const
Pointer points directly to a block.
const Descriptor * getDeclDesc() const
Accessor for information about the declaration site.
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.
static bool hasSameArray(const Pointer &A, const Pointer &B)
Checks if two pointers can be subtracted.
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.
bool isBlockPointer() const
std::optional< APValue > toRValue(const Context &Ctx, QualType ResultType) const
Converts the pointer to an APValue that is an rvalue.
const FunctionPointer & asFunctionPointer() const
bool allElementsInitialized() const
const Block * block() const
bool isFunctionPointer() const
void endLifetime() const
Ends the lifetime of the pointer.
Pointer getDeclPtr() const
const Descriptor * getFieldDesc() const
Accessors for information about the innermost field.
bool isVirtualBaseClass() const
bool isBaseClass() const
Checks if a structure is a base class.
size_t elemSize() const
Returns the element size of the innermost field.
Lifetime getLifetime() const
const BlockPointer & asBlockPointer() const
void initialize() const
Initializes a field.
bool isField() const
Checks if the item is a field in an object.
void initializeElement(unsigned Index) const
Initialized the given element of a primitive array.
const Record * getRecord() const
Returns the record descriptor of a class.
Structure/Class descriptor.
const RecordDecl * getDecl() const
Returns the underlying declaration.
bool isUnion() const
Checks if the record is a union.
const Field * getField(unsigned I) const
unsigned getNumBases() const
const Base * getBase(unsigned I) const
const Base * getVirtualBase(unsigned I) const
unsigned getNumFields() const
unsigned getNumVirtualBases() const
llvm::iterator_range< const_field_iter > fields() const
constexpr bool isIntegerOrBoolType(PrimType T)
PrimType
Enumeration of the primitive types of the VM.
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
@ Result
The result type of a method or function.
U cast(CodeGen::Address addr)
int const char * function
__UINTPTR_TYPE__ uintptr_t
An unsigned integer type with the property that any valid pointer to void can be converted to this ty...
unsigned Base
Start of the current subfield.
Block * Pointee
The block the pointer is pointing to.
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
const ValueDecl * asValueDecl() const
const Decl * asDecl() const
unsigned getMetadataSize() const
Returns the size of the metadata.
QualType getDataType(const ASTContext &Ctx) const
const bool IsArray
Flag indicating if the block is an array.
bool isPrimitiveArray() const
Checks if the descriptor is of an array of primitives.
const FieldDecl * asFieldDecl() const
PrimType getPrimType() const
const Expr * asExpr() const
Descriptor used for global variables.
GlobalInitState InitState
A pointer-sized struct we use to allocate into data storage.
void noteAllInitialized()
bool allInitialized() const
Are all elements in the array already initialized?
void setInitMap(const InitMap *IM)
Bitfield tracking the initialisation status of elements of primitive arrays.
void startElementLifetime(unsigned I)
void endElementLifetime(unsigned I)
bool allElementsAlive() const
bool isElementInitialized(unsigned I) const
Checks if an element was initialized.
bool initializeElement(unsigned I)
Initializes an element. Returns true when object if fully initialized.
bool isElementAlive(unsigned I) const
Inline descriptor embedded in structures and arrays.
unsigned IsActive
Flag indicating if the field is the active member of a union.
unsigned Offset
Offset inside the structure/array.
std::optional< IntPointer > atOffset(const ASTContext &ASTCtx, unsigned Offset) const
IntPointer baseCast(const ASTContext &ASTCtx, unsigned BaseOffset) const
const Type * TypeInfoType