1 //===--- APValue.h - Union class for APFloat/APSInt/Complex -----*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 // 10 // This file defines the APValue class. 11 // 12 //===----------------------------------------------------------------------===// 13 14 #ifndef LLVM_CLANG_AST_APVALUE_H 15 #define LLVM_CLANG_AST_APVALUE_H 16 17 #include "clang/Basic/LLVM.h" 18 #include "llvm/ADT/APFloat.h" 19 #include "llvm/ADT/APSInt.h" 20 #include "llvm/ADT/PointerIntPair.h" 21 #include "llvm/ADT/PointerUnion.h" 22 23 namespace clang { 24 class AddrLabelExpr; 25 class ASTContext; 26 class CharUnits; 27 class DiagnosticBuilder; 28 class Expr; 29 class FieldDecl; 30 class Decl; 31 class ValueDecl; 32 class CXXRecordDecl; 33 class QualType; 34 35 /// APValue - This class implements a discriminated union of [uninitialized] 36 /// [APSInt] [APFloat], [Complex APSInt] [Complex APFloat], [Expr + Offset], 37 /// [Vector: N * APValue], [Array: N * APValue] 38 class APValue { 39 typedef llvm::APSInt APSInt; 40 typedef llvm::APFloat APFloat; 41 public: 42 enum ValueKind { 43 Uninitialized, 44 Int, 45 Float, 46 ComplexInt, 47 ComplexFloat, 48 LValue, 49 Vector, 50 Array, 51 Struct, 52 Union, 53 MemberPointer, 54 AddrLabelDiff 55 }; 56 typedef llvm::PointerUnion<const ValueDecl *, const Expr *> LValueBase; 57 typedef llvm::PointerIntPair<const Decl *, 1, bool> BaseOrMemberType; 58 union LValuePathEntry { 59 /// BaseOrMember - The FieldDecl or CXXRecordDecl indicating the next item 60 /// in the path. An opaque value of type BaseOrMemberType. 61 void *BaseOrMember; 62 /// ArrayIndex - The array index of the next item in the path. 63 uint64_t ArrayIndex; 64 }; 65 struct NoLValuePath {}; 66 struct UninitArray {}; 67 struct UninitStruct {}; 68 private: 69 ValueKind Kind; 70 71 struct ComplexAPSInt { 72 APSInt Real, Imag; ComplexAPSIntComplexAPSInt73 ComplexAPSInt() : Real(1), Imag(1) {} 74 }; 75 struct ComplexAPFloat { 76 APFloat Real, Imag; ComplexAPFloatComplexAPFloat77 ComplexAPFloat() : Real(0.0), Imag(0.0) {} 78 }; 79 struct LV; 80 struct Vec { 81 APValue *Elts; 82 unsigned NumElts; VecVec83 Vec() : Elts(0), NumElts(0) {} ~VecVec84 ~Vec() { delete[] Elts; } 85 }; 86 struct Arr { 87 APValue *Elts; 88 unsigned NumElts, ArrSize; 89 Arr(unsigned NumElts, unsigned ArrSize); 90 ~Arr(); 91 }; 92 struct StructData { 93 APValue *Elts; 94 unsigned NumBases; 95 unsigned NumFields; 96 StructData(unsigned NumBases, unsigned NumFields); 97 ~StructData(); 98 }; 99 struct UnionData { 100 const FieldDecl *Field; 101 APValue *Value; 102 UnionData(); 103 ~UnionData(); 104 }; 105 struct AddrLabelDiffData { 106 const AddrLabelExpr* LHSExpr; 107 const AddrLabelExpr* RHSExpr; 108 }; 109 struct MemberPointerData; 110 111 enum { 112 MaxSize = (sizeof(ComplexAPSInt) > sizeof(ComplexAPFloat) ? 113 sizeof(ComplexAPSInt) : sizeof(ComplexAPFloat)) 114 }; 115 116 union { 117 void *Aligner; 118 char Data[MaxSize]; 119 }; 120 121 public: APValue()122 APValue() : Kind(Uninitialized) {} APValue(const APSInt & I)123 explicit APValue(const APSInt &I) : Kind(Uninitialized) { 124 MakeInt(); setInt(I); 125 } APValue(const APFloat & F)126 explicit APValue(const APFloat &F) : Kind(Uninitialized) { 127 MakeFloat(); setFloat(F); 128 } APValue(const APValue * E,unsigned N)129 explicit APValue(const APValue *E, unsigned N) : Kind(Uninitialized) { 130 MakeVector(); setVector(E, N); 131 } APValue(const APSInt & R,const APSInt & I)132 APValue(const APSInt &R, const APSInt &I) : Kind(Uninitialized) { 133 MakeComplexInt(); setComplexInt(R, I); 134 } APValue(const APFloat & R,const APFloat & I)135 APValue(const APFloat &R, const APFloat &I) : Kind(Uninitialized) { 136 MakeComplexFloat(); setComplexFloat(R, I); 137 } 138 APValue(const APValue &RHS); APValue(LValueBase B,const CharUnits & O,NoLValuePath N,unsigned CallIndex)139 APValue(LValueBase B, const CharUnits &O, NoLValuePath N, unsigned CallIndex) 140 : Kind(Uninitialized) { 141 MakeLValue(); setLValue(B, O, N, CallIndex); 142 } APValue(LValueBase B,const CharUnits & O,ArrayRef<LValuePathEntry> Path,bool OnePastTheEnd,unsigned CallIndex)143 APValue(LValueBase B, const CharUnits &O, ArrayRef<LValuePathEntry> Path, 144 bool OnePastTheEnd, unsigned CallIndex) 145 : Kind(Uninitialized) { 146 MakeLValue(); setLValue(B, O, Path, OnePastTheEnd, CallIndex); 147 } APValue(UninitArray,unsigned InitElts,unsigned Size)148 APValue(UninitArray, unsigned InitElts, unsigned Size) : Kind(Uninitialized) { 149 MakeArray(InitElts, Size); 150 } APValue(UninitStruct,unsigned B,unsigned M)151 APValue(UninitStruct, unsigned B, unsigned M) : Kind(Uninitialized) { 152 MakeStruct(B, M); 153 } 154 explicit APValue(const FieldDecl *D, const APValue &V = APValue()) Kind(Uninitialized)155 : Kind(Uninitialized) { 156 MakeUnion(); setUnion(D, V); 157 } APValue(const ValueDecl * Member,bool IsDerivedMember,ArrayRef<const CXXRecordDecl * > Path)158 APValue(const ValueDecl *Member, bool IsDerivedMember, 159 ArrayRef<const CXXRecordDecl*> Path) : Kind(Uninitialized) { 160 MakeMemberPointer(Member, IsDerivedMember, Path); 161 } APValue(const AddrLabelExpr * LHSExpr,const AddrLabelExpr * RHSExpr)162 APValue(const AddrLabelExpr* LHSExpr, const AddrLabelExpr* RHSExpr) 163 : Kind(Uninitialized) { 164 MakeAddrLabelDiff(); setAddrLabelDiff(LHSExpr, RHSExpr); 165 } 166 ~APValue()167 ~APValue() { 168 MakeUninit(); 169 } 170 171 /// \brief Returns whether the object performed allocations. 172 /// 173 /// If APValues are constructed via placement new, \c needsCleanup() 174 /// indicates whether the destructor must be called in order to correctly 175 /// free all allocated memory. 176 bool needsCleanup() const; 177 178 /// \brief Swaps the contents of this and the given APValue. 179 void swap(APValue &RHS); 180 getKind()181 ValueKind getKind() const { return Kind; } isUninit()182 bool isUninit() const { return Kind == Uninitialized; } isInt()183 bool isInt() const { return Kind == Int; } isFloat()184 bool isFloat() const { return Kind == Float; } isComplexInt()185 bool isComplexInt() const { return Kind == ComplexInt; } isComplexFloat()186 bool isComplexFloat() const { return Kind == ComplexFloat; } isLValue()187 bool isLValue() const { return Kind == LValue; } isVector()188 bool isVector() const { return Kind == Vector; } isArray()189 bool isArray() const { return Kind == Array; } isStruct()190 bool isStruct() const { return Kind == Struct; } isUnion()191 bool isUnion() const { return Kind == Union; } isMemberPointer()192 bool isMemberPointer() const { return Kind == MemberPointer; } isAddrLabelDiff()193 bool isAddrLabelDiff() const { return Kind == AddrLabelDiff; } 194 195 void dump() const; 196 void dump(raw_ostream &OS) const; 197 198 void printPretty(raw_ostream &OS, ASTContext &Ctx, QualType Ty) const; 199 std::string getAsString(ASTContext &Ctx, QualType Ty) const; 200 getInt()201 APSInt &getInt() { 202 assert(isInt() && "Invalid accessor"); 203 return *(APSInt*)(char*)Data; 204 } getInt()205 const APSInt &getInt() const { 206 return const_cast<APValue*>(this)->getInt(); 207 } 208 getFloat()209 APFloat &getFloat() { 210 assert(isFloat() && "Invalid accessor"); 211 return *(APFloat*)(char*)Data; 212 } getFloat()213 const APFloat &getFloat() const { 214 return const_cast<APValue*>(this)->getFloat(); 215 } 216 getComplexIntReal()217 APSInt &getComplexIntReal() { 218 assert(isComplexInt() && "Invalid accessor"); 219 return ((ComplexAPSInt*)(char*)Data)->Real; 220 } getComplexIntReal()221 const APSInt &getComplexIntReal() const { 222 return const_cast<APValue*>(this)->getComplexIntReal(); 223 } 224 getComplexIntImag()225 APSInt &getComplexIntImag() { 226 assert(isComplexInt() && "Invalid accessor"); 227 return ((ComplexAPSInt*)(char*)Data)->Imag; 228 } getComplexIntImag()229 const APSInt &getComplexIntImag() const { 230 return const_cast<APValue*>(this)->getComplexIntImag(); 231 } 232 getComplexFloatReal()233 APFloat &getComplexFloatReal() { 234 assert(isComplexFloat() && "Invalid accessor"); 235 return ((ComplexAPFloat*)(char*)Data)->Real; 236 } getComplexFloatReal()237 const APFloat &getComplexFloatReal() const { 238 return const_cast<APValue*>(this)->getComplexFloatReal(); 239 } 240 getComplexFloatImag()241 APFloat &getComplexFloatImag() { 242 assert(isComplexFloat() && "Invalid accessor"); 243 return ((ComplexAPFloat*)(char*)Data)->Imag; 244 } getComplexFloatImag()245 const APFloat &getComplexFloatImag() const { 246 return const_cast<APValue*>(this)->getComplexFloatImag(); 247 } 248 249 const LValueBase getLValueBase() const; 250 CharUnits &getLValueOffset(); getLValueOffset()251 const CharUnits &getLValueOffset() const { 252 return const_cast<APValue*>(this)->getLValueOffset(); 253 } 254 bool isLValueOnePastTheEnd() const; 255 bool hasLValuePath() const; 256 ArrayRef<LValuePathEntry> getLValuePath() const; 257 unsigned getLValueCallIndex() const; 258 getVectorElt(unsigned I)259 APValue &getVectorElt(unsigned I) { 260 assert(isVector() && "Invalid accessor"); 261 assert(I < getVectorLength() && "Index out of range"); 262 return ((Vec*)(char*)Data)->Elts[I]; 263 } getVectorElt(unsigned I)264 const APValue &getVectorElt(unsigned I) const { 265 return const_cast<APValue*>(this)->getVectorElt(I); 266 } getVectorLength()267 unsigned getVectorLength() const { 268 assert(isVector() && "Invalid accessor"); 269 return ((const Vec*)(const void *)Data)->NumElts; 270 } 271 getArrayInitializedElt(unsigned I)272 APValue &getArrayInitializedElt(unsigned I) { 273 assert(isArray() && "Invalid accessor"); 274 assert(I < getArrayInitializedElts() && "Index out of range"); 275 return ((Arr*)(char*)Data)->Elts[I]; 276 } getArrayInitializedElt(unsigned I)277 const APValue &getArrayInitializedElt(unsigned I) const { 278 return const_cast<APValue*>(this)->getArrayInitializedElt(I); 279 } hasArrayFiller()280 bool hasArrayFiller() const { 281 return getArrayInitializedElts() != getArraySize(); 282 } getArrayFiller()283 APValue &getArrayFiller() { 284 assert(isArray() && "Invalid accessor"); 285 assert(hasArrayFiller() && "No array filler"); 286 return ((Arr*)(char*)Data)->Elts[getArrayInitializedElts()]; 287 } getArrayFiller()288 const APValue &getArrayFiller() const { 289 return const_cast<APValue*>(this)->getArrayFiller(); 290 } getArrayInitializedElts()291 unsigned getArrayInitializedElts() const { 292 assert(isArray() && "Invalid accessor"); 293 return ((const Arr*)(const void *)Data)->NumElts; 294 } getArraySize()295 unsigned getArraySize() const { 296 assert(isArray() && "Invalid accessor"); 297 return ((const Arr*)(const void *)Data)->ArrSize; 298 } 299 getStructNumBases()300 unsigned getStructNumBases() const { 301 assert(isStruct() && "Invalid accessor"); 302 return ((const StructData*)(const char*)Data)->NumBases; 303 } getStructNumFields()304 unsigned getStructNumFields() const { 305 assert(isStruct() && "Invalid accessor"); 306 return ((const StructData*)(const char*)Data)->NumFields; 307 } getStructBase(unsigned i)308 APValue &getStructBase(unsigned i) { 309 assert(isStruct() && "Invalid accessor"); 310 return ((StructData*)(char*)Data)->Elts[i]; 311 } getStructField(unsigned i)312 APValue &getStructField(unsigned i) { 313 assert(isStruct() && "Invalid accessor"); 314 return ((StructData*)(char*)Data)->Elts[getStructNumBases() + i]; 315 } getStructBase(unsigned i)316 const APValue &getStructBase(unsigned i) const { 317 return const_cast<APValue*>(this)->getStructBase(i); 318 } getStructField(unsigned i)319 const APValue &getStructField(unsigned i) const { 320 return const_cast<APValue*>(this)->getStructField(i); 321 } 322 getUnionField()323 const FieldDecl *getUnionField() const { 324 assert(isUnion() && "Invalid accessor"); 325 return ((const UnionData*)(const char*)Data)->Field; 326 } getUnionValue()327 APValue &getUnionValue() { 328 assert(isUnion() && "Invalid accessor"); 329 return *((UnionData*)(char*)Data)->Value; 330 } getUnionValue()331 const APValue &getUnionValue() const { 332 return const_cast<APValue*>(this)->getUnionValue(); 333 } 334 335 const ValueDecl *getMemberPointerDecl() const; 336 bool isMemberPointerToDerivedMember() const; 337 ArrayRef<const CXXRecordDecl*> getMemberPointerPath() const; 338 getAddrLabelDiffLHS()339 const AddrLabelExpr* getAddrLabelDiffLHS() const { 340 assert(isAddrLabelDiff() && "Invalid accessor"); 341 return ((const AddrLabelDiffData*)(const char*)Data)->LHSExpr; 342 } getAddrLabelDiffRHS()343 const AddrLabelExpr* getAddrLabelDiffRHS() const { 344 assert(isAddrLabelDiff() && "Invalid accessor"); 345 return ((const AddrLabelDiffData*)(const char*)Data)->RHSExpr; 346 } 347 setInt(const APSInt & I)348 void setInt(const APSInt &I) { 349 assert(isInt() && "Invalid accessor"); 350 *(APSInt*)(char*)Data = I; 351 } setFloat(const APFloat & F)352 void setFloat(const APFloat &F) { 353 assert(isFloat() && "Invalid accessor"); 354 *(APFloat*)(char*)Data = F; 355 } setVector(const APValue * E,unsigned N)356 void setVector(const APValue *E, unsigned N) { 357 assert(isVector() && "Invalid accessor"); 358 ((Vec*)(char*)Data)->Elts = new APValue[N]; 359 ((Vec*)(char*)Data)->NumElts = N; 360 for (unsigned i = 0; i != N; ++i) 361 ((Vec*)(char*)Data)->Elts[i] = E[i]; 362 } setComplexInt(const APSInt & R,const APSInt & I)363 void setComplexInt(const APSInt &R, const APSInt &I) { 364 assert(R.getBitWidth() == I.getBitWidth() && 365 "Invalid complex int (type mismatch)."); 366 assert(isComplexInt() && "Invalid accessor"); 367 ((ComplexAPSInt*)(char*)Data)->Real = R; 368 ((ComplexAPSInt*)(char*)Data)->Imag = I; 369 } setComplexFloat(const APFloat & R,const APFloat & I)370 void setComplexFloat(const APFloat &R, const APFloat &I) { 371 assert(&R.getSemantics() == &I.getSemantics() && 372 "Invalid complex float (type mismatch)."); 373 assert(isComplexFloat() && "Invalid accessor"); 374 ((ComplexAPFloat*)(char*)Data)->Real = R; 375 ((ComplexAPFloat*)(char*)Data)->Imag = I; 376 } 377 void setLValue(LValueBase B, const CharUnits &O, NoLValuePath, 378 unsigned CallIndex); 379 void setLValue(LValueBase B, const CharUnits &O, 380 ArrayRef<LValuePathEntry> Path, bool OnePastTheEnd, 381 unsigned CallIndex); setUnion(const FieldDecl * Field,const APValue & Value)382 void setUnion(const FieldDecl *Field, const APValue &Value) { 383 assert(isUnion() && "Invalid accessor"); 384 ((UnionData*)(char*)Data)->Field = Field; 385 *((UnionData*)(char*)Data)->Value = Value; 386 } setAddrLabelDiff(const AddrLabelExpr * LHSExpr,const AddrLabelExpr * RHSExpr)387 void setAddrLabelDiff(const AddrLabelExpr* LHSExpr, 388 const AddrLabelExpr* RHSExpr) { 389 ((AddrLabelDiffData*)(char*)Data)->LHSExpr = LHSExpr; 390 ((AddrLabelDiffData*)(char*)Data)->RHSExpr = RHSExpr; 391 } 392 393 /// Assign by swapping from a copy of the RHS. 394 APValue &operator=(APValue RHS) { 395 swap(RHS); 396 return *this; 397 } 398 399 private: 400 void DestroyDataAndMakeUninit(); MakeUninit()401 void MakeUninit() { 402 if (Kind != Uninitialized) 403 DestroyDataAndMakeUninit(); 404 } MakeInt()405 void MakeInt() { 406 assert(isUninit() && "Bad state change"); 407 new ((void*)Data) APSInt(1); 408 Kind = Int; 409 } MakeFloat()410 void MakeFloat() { 411 assert(isUninit() && "Bad state change"); 412 new ((void*)(char*)Data) APFloat(0.0); 413 Kind = Float; 414 } MakeVector()415 void MakeVector() { 416 assert(isUninit() && "Bad state change"); 417 new ((void*)(char*)Data) Vec(); 418 Kind = Vector; 419 } MakeComplexInt()420 void MakeComplexInt() { 421 assert(isUninit() && "Bad state change"); 422 new ((void*)(char*)Data) ComplexAPSInt(); 423 Kind = ComplexInt; 424 } MakeComplexFloat()425 void MakeComplexFloat() { 426 assert(isUninit() && "Bad state change"); 427 new ((void*)(char*)Data) ComplexAPFloat(); 428 Kind = ComplexFloat; 429 } 430 void MakeLValue(); 431 void MakeArray(unsigned InitElts, unsigned Size); MakeStruct(unsigned B,unsigned M)432 void MakeStruct(unsigned B, unsigned M) { 433 assert(isUninit() && "Bad state change"); 434 new ((void*)(char*)Data) StructData(B, M); 435 Kind = Struct; 436 } MakeUnion()437 void MakeUnion() { 438 assert(isUninit() && "Bad state change"); 439 new ((void*)(char*)Data) UnionData(); 440 Kind = Union; 441 } 442 void MakeMemberPointer(const ValueDecl *Member, bool IsDerivedMember, 443 ArrayRef<const CXXRecordDecl*> Path); MakeAddrLabelDiff()444 void MakeAddrLabelDiff() { 445 assert(isUninit() && "Bad state change"); 446 new ((void*)(char*)Data) AddrLabelDiffData(); 447 Kind = AddrLabelDiff; 448 } 449 }; 450 451 } // end namespace clang. 452 453 #endif 454