| Anders Carlsson | 5b95592 | 2009-11-24 05:51:11 +0000 | [diff] [blame] | 1 | //===--- CGClass.cpp - Emit LLVM Code for C++ classes ---------------------===// | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 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 contains code dealing with C++ code generation of classes | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
|  | 14 | #include "CodeGenFunction.h" | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 15 | #include "clang/AST/CXXInheritance.h" | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 16 | #include "clang/AST/RecordLayout.h" | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 17 | #include "clang/AST/StmtCXX.h" | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 18 |  | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 19 | using namespace clang; | 
|  | 20 | using namespace CodeGen; | 
|  | 21 |  | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 22 | static uint64_t | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 23 | ComputeNonVirtualBaseClassOffset(ASTContext &Context, | 
|  | 24 | const CXXRecordDecl *DerivedClass, | 
|  | 25 | CXXBaseSpecifierArray::iterator Start, | 
|  | 26 | CXXBaseSpecifierArray::iterator End) { | 
|  | 27 | uint64_t Offset = 0; | 
|  | 28 |  | 
|  | 29 | const CXXRecordDecl *RD = DerivedClass; | 
|  | 30 |  | 
|  | 31 | for (CXXBaseSpecifierArray::iterator I = Start; I != End; ++I) { | 
|  | 32 | const CXXBaseSpecifier *Base = *I; | 
|  | 33 | assert(!Base->isVirtual() && "Should not see virtual bases here!"); | 
|  | 34 |  | 
|  | 35 | // Get the layout. | 
|  | 36 | const ASTRecordLayout &Layout = Context.getASTRecordLayout(RD); | 
|  | 37 |  | 
|  | 38 | const CXXRecordDecl *BaseDecl = | 
|  | 39 | cast<CXXRecordDecl>(Base->getType()->getAs<RecordType>()->getDecl()); | 
|  | 40 |  | 
|  | 41 | // Add the offset. | 
|  | 42 | Offset += Layout.getBaseClassOffset(BaseDecl); | 
|  | 43 |  | 
|  | 44 | RD = BaseDecl; | 
|  | 45 | } | 
|  | 46 |  | 
|  | 47 | // FIXME: We should not use / 8 here. | 
|  | 48 | return Offset / 8; | 
|  | 49 | } | 
|  | 50 |  | 
|  | 51 | static uint64_t | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 52 | ComputeNonVirtualBaseClassOffset(ASTContext &Context, | 
|  | 53 | const CXXBasePath &Path, | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 54 | unsigned Start) { | 
|  | 55 | uint64_t Offset = 0; | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 56 |  | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 57 | for (unsigned i = Start, e = Path.size(); i != e; ++i) { | 
|  | 58 | const CXXBasePathElement& Element = Path[i]; | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 59 |  | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 60 | // Get the layout. | 
|  | 61 | const ASTRecordLayout &Layout = Context.getASTRecordLayout(Element.Class); | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 62 |  | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 63 | const CXXBaseSpecifier *BS = Element.Base; | 
|  | 64 | assert(!BS->isVirtual() && "Should not see virtual bases here!"); | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 65 |  | 
| Anders Carlsson | 2f1986b | 2009-10-06 22:43:30 +0000 | [diff] [blame] | 66 | const CXXRecordDecl *Base = | 
|  | 67 | cast<CXXRecordDecl>(BS->getType()->getAs<RecordType>()->getDecl()); | 
|  | 68 |  | 
|  | 69 | // Add the offset. | 
|  | 70 | Offset += Layout.getBaseClassOffset(Base) / 8; | 
|  | 71 | } | 
|  | 72 |  | 
|  | 73 | return Offset; | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 74 | } | 
|  | 75 |  | 
| Anders Carlsson | 84080ec | 2009-09-29 03:13:20 +0000 | [diff] [blame] | 76 | llvm::Constant * | 
| Anders Carlsson | a04efdf | 2010-04-24 21:23:59 +0000 | [diff] [blame] | 77 | CodeGenModule::GetNonVirtualBaseClassOffset(const CXXRecordDecl *ClassDecl, | 
|  | 78 | const CXXBaseSpecifierArray &BasePath) { | 
|  | 79 | assert(!BasePath.empty() && "Base path should not be empty!"); | 
|  | 80 |  | 
|  | 81 | uint64_t Offset = | 
|  | 82 | ComputeNonVirtualBaseClassOffset(getContext(), ClassDecl, | 
|  | 83 | BasePath.begin(), BasePath.end()); | 
|  | 84 | if (!Offset) | 
|  | 85 | return 0; | 
|  | 86 |  | 
|  | 87 | const llvm::Type *PtrDiffTy = | 
|  | 88 | Types.ConvertType(getContext().getPointerDiffType()); | 
|  | 89 |  | 
|  | 90 | return llvm::ConstantInt::get(PtrDiffTy, Offset); | 
| Anders Carlsson | 84080ec | 2009-09-29 03:13:20 +0000 | [diff] [blame] | 91 | } | 
|  | 92 |  | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 93 | /// Gets the address of a direct base class within a complete object. | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 94 | /// This should only be used for (1) non-virtual bases or (2) virtual bases | 
|  | 95 | /// when the type is known to be complete (e.g. in complete destructors). | 
|  | 96 | /// | 
|  | 97 | /// The object pointed to by 'This' is assumed to be non-null. | 
|  | 98 | llvm::Value * | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 99 | CodeGenFunction::GetAddressOfDirectBaseInCompleteClass(llvm::Value *This, | 
|  | 100 | const CXXRecordDecl *Derived, | 
|  | 101 | const CXXRecordDecl *Base, | 
|  | 102 | bool BaseIsVirtual) { | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 103 | // 'this' must be a pointer (in some address space) to Derived. | 
|  | 104 | assert(This->getType()->isPointerTy() && | 
|  | 105 | cast<llvm::PointerType>(This->getType())->getElementType() | 
|  | 106 | == ConvertType(Derived)); | 
|  | 107 |  | 
|  | 108 | // Compute the offset of the virtual base. | 
|  | 109 | uint64_t Offset; | 
|  | 110 | const ASTRecordLayout &Layout = getContext().getASTRecordLayout(Derived); | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 111 | if (BaseIsVirtual) | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 112 | Offset = Layout.getVBaseClassOffset(Base); | 
|  | 113 | else | 
|  | 114 | Offset = Layout.getBaseClassOffset(Base); | 
|  | 115 |  | 
|  | 116 | // Shift and cast down to the base type. | 
|  | 117 | // TODO: for complete types, this should be possible with a GEP. | 
|  | 118 | llvm::Value *V = This; | 
|  | 119 | if (Offset) { | 
|  | 120 | const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(getLLVMContext()); | 
|  | 121 | V = Builder.CreateBitCast(V, Int8PtrTy); | 
|  | 122 | V = Builder.CreateConstInBoundsGEP1_64(V, Offset / 8); | 
|  | 123 | } | 
|  | 124 | V = Builder.CreateBitCast(V, ConvertType(Base)->getPointerTo()); | 
|  | 125 |  | 
|  | 126 | return V; | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 127 | } | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 128 |  | 
| Anders Carlsson | 9dc228a | 2010-04-20 16:03:35 +0000 | [diff] [blame] | 129 | static llvm::Value * | 
|  | 130 | ApplyNonVirtualAndVirtualOffset(CodeGenFunction &CGF, llvm::Value *ThisPtr, | 
|  | 131 | uint64_t NonVirtual, llvm::Value *Virtual) { | 
|  | 132 | const llvm::Type *PtrDiffTy = | 
|  | 133 | CGF.ConvertType(CGF.getContext().getPointerDiffType()); | 
|  | 134 |  | 
|  | 135 | llvm::Value *NonVirtualOffset = 0; | 
|  | 136 | if (NonVirtual) | 
|  | 137 | NonVirtualOffset = llvm::ConstantInt::get(PtrDiffTy, NonVirtual); | 
|  | 138 |  | 
|  | 139 | llvm::Value *BaseOffset; | 
|  | 140 | if (Virtual) { | 
|  | 141 | if (NonVirtualOffset) | 
|  | 142 | BaseOffset = CGF.Builder.CreateAdd(Virtual, NonVirtualOffset); | 
|  | 143 | else | 
|  | 144 | BaseOffset = Virtual; | 
|  | 145 | } else | 
|  | 146 | BaseOffset = NonVirtualOffset; | 
|  | 147 |  | 
|  | 148 | // Apply the base offset. | 
|  | 149 | const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGF.getLLVMContext()); | 
|  | 150 | ThisPtr = CGF.Builder.CreateBitCast(ThisPtr, Int8PtrTy); | 
|  | 151 | ThisPtr = CGF.Builder.CreateGEP(ThisPtr, BaseOffset, "add.ptr"); | 
|  | 152 |  | 
|  | 153 | return ThisPtr; | 
|  | 154 | } | 
|  | 155 |  | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 156 | llvm::Value * | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 157 | CodeGenFunction::GetAddressOfBaseClass(llvm::Value *Value, | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 158 | const CXXRecordDecl *Derived, | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 159 | const CXXBaseSpecifierArray &BasePath, | 
|  | 160 | bool NullCheckValue) { | 
|  | 161 | assert(!BasePath.empty() && "Base path should not be empty!"); | 
|  | 162 |  | 
|  | 163 | CXXBaseSpecifierArray::iterator Start = BasePath.begin(); | 
|  | 164 | const CXXRecordDecl *VBase = 0; | 
|  | 165 |  | 
|  | 166 | // Get the virtual base. | 
|  | 167 | if ((*Start)->isVirtual()) { | 
|  | 168 | VBase = | 
|  | 169 | cast<CXXRecordDecl>((*Start)->getType()->getAs<RecordType>()->getDecl()); | 
|  | 170 | ++Start; | 
|  | 171 | } | 
|  | 172 |  | 
|  | 173 | uint64_t NonVirtualOffset = | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 174 | ComputeNonVirtualBaseClassOffset(getContext(), VBase ? VBase : Derived, | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 175 | Start, BasePath.end()); | 
|  | 176 |  | 
|  | 177 | // Get the base pointer type. | 
|  | 178 | const llvm::Type *BasePtrTy = | 
| Anders Carlsson | fc89c31 | 2010-04-24 21:12:55 +0000 | [diff] [blame] | 179 | ConvertType((BasePath.end()[-1])->getType())->getPointerTo(); | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 180 |  | 
|  | 181 | if (!NonVirtualOffset && !VBase) { | 
|  | 182 | // Just cast back. | 
|  | 183 | return Builder.CreateBitCast(Value, BasePtrTy); | 
|  | 184 | } | 
|  | 185 |  | 
|  | 186 | llvm::BasicBlock *CastNull = 0; | 
|  | 187 | llvm::BasicBlock *CastNotNull = 0; | 
|  | 188 | llvm::BasicBlock *CastEnd = 0; | 
|  | 189 |  | 
|  | 190 | if (NullCheckValue) { | 
|  | 191 | CastNull = createBasicBlock("cast.null"); | 
|  | 192 | CastNotNull = createBasicBlock("cast.notnull"); | 
|  | 193 | CastEnd = createBasicBlock("cast.end"); | 
|  | 194 |  | 
|  | 195 | llvm::Value *IsNull = | 
|  | 196 | Builder.CreateICmpEQ(Value, | 
|  | 197 | llvm::Constant::getNullValue(Value->getType())); | 
|  | 198 | Builder.CreateCondBr(IsNull, CastNull, CastNotNull); | 
|  | 199 | EmitBlock(CastNotNull); | 
|  | 200 | } | 
|  | 201 |  | 
|  | 202 | llvm::Value *VirtualOffset = 0; | 
|  | 203 |  | 
|  | 204 | if (VBase) | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 205 | VirtualOffset = GetVirtualBaseClassOffset(Value, Derived, VBase); | 
| Anders Carlsson | 34a2d38 | 2010-04-24 21:06:20 +0000 | [diff] [blame] | 206 |  | 
|  | 207 | // Apply the offsets. | 
|  | 208 | Value = ApplyNonVirtualAndVirtualOffset(*this, Value, NonVirtualOffset, | 
|  | 209 | VirtualOffset); | 
|  | 210 |  | 
|  | 211 | // Cast back. | 
|  | 212 | Value = Builder.CreateBitCast(Value, BasePtrTy); | 
|  | 213 |  | 
|  | 214 | if (NullCheckValue) { | 
|  | 215 | Builder.CreateBr(CastEnd); | 
|  | 216 | EmitBlock(CastNull); | 
|  | 217 | Builder.CreateBr(CastEnd); | 
|  | 218 | EmitBlock(CastEnd); | 
|  | 219 |  | 
|  | 220 | llvm::PHINode *PHI = Builder.CreatePHI(Value->getType()); | 
|  | 221 | PHI->reserveOperandSpace(2); | 
|  | 222 | PHI->addIncoming(Value, CastNotNull); | 
|  | 223 | PHI->addIncoming(llvm::Constant::getNullValue(Value->getType()), | 
|  | 224 | CastNull); | 
|  | 225 | Value = PHI; | 
|  | 226 | } | 
|  | 227 |  | 
|  | 228 | return Value; | 
|  | 229 | } | 
|  | 230 |  | 
|  | 231 | llvm::Value * | 
| Anders Carlsson | a88ad56 | 2010-04-24 21:51:08 +0000 | [diff] [blame] | 232 | CodeGenFunction::OldGetAddressOfBaseClass(llvm::Value *Value, | 
|  | 233 | const CXXRecordDecl *Class, | 
|  | 234 | const CXXRecordDecl *BaseClass) { | 
| Anders Carlsson | dfd0330 | 2009-09-22 21:58:22 +0000 | [diff] [blame] | 235 | QualType BTy = | 
|  | 236 | getContext().getCanonicalType( | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 237 | getContext().getTypeDeclType(BaseClass)); | 
| Anders Carlsson | dfd0330 | 2009-09-22 21:58:22 +0000 | [diff] [blame] | 238 | const llvm::Type *BasePtrTy = llvm::PointerType::getUnqual(ConvertType(BTy)); | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 239 |  | 
| Anders Carlsson | bb7e17b | 2010-01-31 01:36:53 +0000 | [diff] [blame] | 240 | if (Class == BaseClass) { | 
| Anders Carlsson | dfd0330 | 2009-09-22 21:58:22 +0000 | [diff] [blame] | 241 | // Just cast back. | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 242 | return Builder.CreateBitCast(Value, BasePtrTy); | 
| Anders Carlsson | dfd0330 | 2009-09-22 21:58:22 +0000 | [diff] [blame] | 243 | } | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 244 |  | 
| Anders Carlsson | 2692d82 | 2010-04-20 05:07:22 +0000 | [diff] [blame] | 245 | #ifndef NDEBUG | 
|  | 246 | CXXBasePaths Paths(/*FindAmbiguities=*/true, | 
|  | 247 | /*RecordPaths=*/true, /*DetectVirtual=*/false); | 
|  | 248 | #else | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 249 | CXXBasePaths Paths(/*FindAmbiguities=*/false, | 
|  | 250 | /*RecordPaths=*/true, /*DetectVirtual=*/false); | 
| Anders Carlsson | 2692d82 | 2010-04-20 05:07:22 +0000 | [diff] [blame] | 251 | #endif | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 252 | if (!const_cast<CXXRecordDecl *>(Class)-> | 
|  | 253 | isDerivedFrom(const_cast<CXXRecordDecl *>(BaseClass), Paths)) { | 
|  | 254 | assert(false && "Class must be derived from the passed in base class!"); | 
|  | 255 | return 0; | 
|  | 256 | } | 
|  | 257 |  | 
| Anders Carlsson | c2a9b79 | 2010-04-21 18:03:05 +0000 | [diff] [blame] | 258 | #if 0 | 
|  | 259 | // FIXME: Re-enable this assert when the underlying bugs have been fixed. | 
| Anders Carlsson | 2692d82 | 2010-04-20 05:07:22 +0000 | [diff] [blame] | 260 | assert(!Paths.isAmbiguous(BTy) && "Path is ambiguous"); | 
| Anders Carlsson | c2a9b79 | 2010-04-21 18:03:05 +0000 | [diff] [blame] | 261 | #endif | 
| Anders Carlsson | 2692d82 | 2010-04-20 05:07:22 +0000 | [diff] [blame] | 262 |  | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 263 | unsigned Start = 0; | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 264 |  | 
|  | 265 | const CXXBasePath &Path = Paths.front(); | 
|  | 266 | const CXXRecordDecl *VBase = 0; | 
|  | 267 | for (unsigned i = 0, e = Path.size(); i != e; ++i) { | 
|  | 268 | const CXXBasePathElement& Element = Path[i]; | 
|  | 269 | if (Element.Base->isVirtual()) { | 
|  | 270 | Start = i+1; | 
|  | 271 | QualType VBaseType = Element.Base->getType(); | 
|  | 272 | VBase = cast<CXXRecordDecl>(VBaseType->getAs<RecordType>()->getDecl()); | 
|  | 273 | } | 
|  | 274 | } | 
|  | 275 |  | 
|  | 276 | uint64_t Offset = | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 277 | ComputeNonVirtualBaseClassOffset(getContext(), Paths.front(), Start); | 
| Eli Friedman | 4a5dc24 | 2009-11-10 22:48:10 +0000 | [diff] [blame] | 278 |  | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 279 | if (!Offset && !VBase) { | 
|  | 280 | // Just cast back. | 
|  | 281 | return Builder.CreateBitCast(Value, BasePtrTy); | 
|  | 282 | } | 
|  | 283 |  | 
| Anders Carlsson | 9dc228a | 2010-04-20 16:03:35 +0000 | [diff] [blame] | 284 | llvm::Value *VirtualOffset = 0; | 
|  | 285 |  | 
| Anders Carlsson | 905a100 | 2010-01-31 02:39:02 +0000 | [diff] [blame] | 286 | if (VBase) | 
|  | 287 | VirtualOffset = GetVirtualBaseClassOffset(Value, Class, VBase); | 
| Eli Friedman | 4a5dc24 | 2009-11-10 22:48:10 +0000 | [diff] [blame] | 288 |  | 
| Anders Carlsson | 9dc228a | 2010-04-20 16:03:35 +0000 | [diff] [blame] | 289 | // Apply the offsets. | 
|  | 290 | Value = ApplyNonVirtualAndVirtualOffset(*this, Value, Offset, VirtualOffset); | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 291 |  | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 292 | // Cast back. | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 293 | Value = Builder.CreateBitCast(Value, BasePtrTy); | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 294 | return Value; | 
|  | 295 | } | 
|  | 296 |  | 
|  | 297 | llvm::Value * | 
|  | 298 | CodeGenFunction::GetAddressOfDerivedClass(llvm::Value *Value, | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 299 | const CXXRecordDecl *Derived, | 
| Anders Carlsson | a04efdf | 2010-04-24 21:23:59 +0000 | [diff] [blame] | 300 | const CXXBaseSpecifierArray &BasePath, | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 301 | bool NullCheckValue) { | 
| Anders Carlsson | a04efdf | 2010-04-24 21:23:59 +0000 | [diff] [blame] | 302 | assert(!BasePath.empty() && "Base path should not be empty!"); | 
|  | 303 |  | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 304 | QualType DerivedTy = | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 305 | getContext().getCanonicalType(getContext().getTagDeclType(Derived)); | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 306 | const llvm::Type *DerivedPtrTy = ConvertType(DerivedTy)->getPointerTo(); | 
|  | 307 |  | 
| Anders Carlsson | a552ea7 | 2010-01-31 01:43:37 +0000 | [diff] [blame] | 308 | llvm::Value *NonVirtualOffset = | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 309 | CGM.GetNonVirtualBaseClassOffset(Derived, BasePath); | 
| Anders Carlsson | a552ea7 | 2010-01-31 01:43:37 +0000 | [diff] [blame] | 310 |  | 
|  | 311 | if (!NonVirtualOffset) { | 
|  | 312 | // No offset, we can just cast back. | 
|  | 313 | return Builder.CreateBitCast(Value, DerivedPtrTy); | 
|  | 314 | } | 
|  | 315 |  | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 316 | llvm::BasicBlock *CastNull = 0; | 
|  | 317 | llvm::BasicBlock *CastNotNull = 0; | 
|  | 318 | llvm::BasicBlock *CastEnd = 0; | 
|  | 319 |  | 
|  | 320 | if (NullCheckValue) { | 
|  | 321 | CastNull = createBasicBlock("cast.null"); | 
|  | 322 | CastNotNull = createBasicBlock("cast.notnull"); | 
|  | 323 | CastEnd = createBasicBlock("cast.end"); | 
|  | 324 |  | 
|  | 325 | llvm::Value *IsNull = | 
|  | 326 | Builder.CreateICmpEQ(Value, | 
|  | 327 | llvm::Constant::getNullValue(Value->getType())); | 
|  | 328 | Builder.CreateCondBr(IsNull, CastNull, CastNotNull); | 
|  | 329 | EmitBlock(CastNotNull); | 
|  | 330 | } | 
|  | 331 |  | 
| Anders Carlsson | a552ea7 | 2010-01-31 01:43:37 +0000 | [diff] [blame] | 332 | // Apply the offset. | 
|  | 333 | Value = Builder.CreatePtrToInt(Value, NonVirtualOffset->getType()); | 
|  | 334 | Value = Builder.CreateSub(Value, NonVirtualOffset); | 
|  | 335 | Value = Builder.CreateIntToPtr(Value, DerivedPtrTy); | 
|  | 336 |  | 
|  | 337 | // Just cast. | 
|  | 338 | Value = Builder.CreateBitCast(Value, DerivedPtrTy); | 
| Anders Carlsson | a3697c9 | 2009-11-23 17:57:54 +0000 | [diff] [blame] | 339 |  | 
|  | 340 | if (NullCheckValue) { | 
|  | 341 | Builder.CreateBr(CastEnd); | 
|  | 342 | EmitBlock(CastNull); | 
|  | 343 | Builder.CreateBr(CastEnd); | 
|  | 344 | EmitBlock(CastEnd); | 
|  | 345 |  | 
|  | 346 | llvm::PHINode *PHI = Builder.CreatePHI(Value->getType()); | 
|  | 347 | PHI->reserveOperandSpace(2); | 
|  | 348 | PHI->addIncoming(Value, CastNotNull); | 
|  | 349 | PHI->addIncoming(llvm::Constant::getNullValue(Value->getType()), | 
|  | 350 | CastNull); | 
|  | 351 | Value = PHI; | 
|  | 352 | } | 
|  | 353 |  | 
|  | 354 | return Value; | 
| Anders Carlsson | 5d58a1d | 2009-09-12 04:27:24 +0000 | [diff] [blame] | 355 | } | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 356 |  | 
| Anders Carlsson | 21c9ad9 | 2010-03-30 03:27:09 +0000 | [diff] [blame] | 357 | /// EmitCopyCtorCall - Emit a call to a copy constructor. | 
|  | 358 | static void | 
| Anders Carlsson | bfe7e91 | 2010-05-01 17:07:40 +0000 | [diff] [blame] | 359 | EmitCopyCtorCall(CodeGenFunction &CGF, const CXXConstructorDecl *CopyCtor, | 
|  | 360 | llvm::Value *ThisPtr, llvm::Value *Src) { | 
|  | 361 | llvm::Value *Callee = CGF.CGM.GetAddrOfCXXConstructor(CopyCtor, Ctor_Complete); | 
| Anders Carlsson | 21c9ad9 | 2010-03-30 03:27:09 +0000 | [diff] [blame] | 362 |  | 
|  | 363 | CallArgList CallArgs; | 
|  | 364 |  | 
|  | 365 | // Push the this ptr. | 
|  | 366 | CallArgs.push_back(std::make_pair(RValue::get(ThisPtr), | 
|  | 367 | CopyCtor->getThisType(CGF.getContext()))); | 
| Anders Carlsson | bfe7e91 | 2010-05-01 17:07:40 +0000 | [diff] [blame] | 368 |  | 
| Anders Carlsson | 21c9ad9 | 2010-03-30 03:27:09 +0000 | [diff] [blame] | 369 | // Push the Src ptr. | 
|  | 370 | CallArgs.push_back(std::make_pair(RValue::get(Src), | 
|  | 371 | CopyCtor->getParamDecl(0)->getType())); | 
|  | 372 |  | 
|  | 373 |  | 
|  | 374 | { | 
|  | 375 | CodeGenFunction::CXXTemporariesCleanupScope Scope(CGF); | 
|  | 376 |  | 
|  | 377 | // If the copy constructor has default arguments, emit them. | 
|  | 378 | for (unsigned I = 1, E = CopyCtor->getNumParams(); I < E; ++I) { | 
|  | 379 | const ParmVarDecl *Param = CopyCtor->getParamDecl(I); | 
|  | 380 | const Expr *DefaultArgExpr = Param->getDefaultArg(); | 
|  | 381 |  | 
|  | 382 | assert(DefaultArgExpr && "Ctor parameter must have default arg!"); | 
|  | 383 |  | 
|  | 384 | QualType ArgType = Param->getType(); | 
|  | 385 | CallArgs.push_back(std::make_pair(CGF.EmitCallArg(DefaultArgExpr, | 
|  | 386 | ArgType), | 
|  | 387 | ArgType)); | 
|  | 388 | } | 
|  | 389 |  | 
|  | 390 | const FunctionProtoType *FPT = | 
|  | 391 | CopyCtor->getType()->getAs<FunctionProtoType>(); | 
|  | 392 | CGF.EmitCall(CGF.CGM.getTypes().getFunctionInfo(CallArgs, FPT), | 
|  | 393 | Callee, ReturnValueSlot(), CallArgs, CopyCtor); | 
|  | 394 | } | 
|  | 395 | } | 
|  | 396 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 397 | /// EmitClassAggrMemberwiseCopy - This routine generates code to copy a class | 
|  | 398 | /// array of objects from SrcValue to DestValue. Copying can be either a bitwise | 
|  | 399 | /// copy or via a copy constructor call. | 
|  | 400 | //  FIXME. Consolidate this with EmitCXXAggrConstructorCall. | 
| Anders Carlsson | 8e142cc | 2010-04-25 00:52:09 +0000 | [diff] [blame] | 401 | void CodeGenFunction::EmitClassAggrMemberwiseCopy(llvm::Value *Dest, | 
|  | 402 | llvm::Value *Src, | 
| Anders Carlsson | 43db20e | 2010-05-01 17:02:18 +0000 | [diff] [blame] | 403 | const ConstantArrayType *Array, | 
|  | 404 | const CXXRecordDecl *ClassDecl) { | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 405 | // Create a temporary for the loop index and initialize it with 0. | 
|  | 406 | llvm::Value *IndexPtr = CreateTempAlloca(llvm::Type::getInt64Ty(VMContext), | 
|  | 407 | "loop.index"); | 
|  | 408 | llvm::Value* zeroConstant = | 
|  | 409 | llvm::Constant::getNullValue(llvm::Type::getInt64Ty(VMContext)); | 
|  | 410 | Builder.CreateStore(zeroConstant, IndexPtr); | 
|  | 411 | // Start the loop with a block that tests the condition. | 
|  | 412 | llvm::BasicBlock *CondBlock = createBasicBlock("for.cond"); | 
|  | 413 | llvm::BasicBlock *AfterFor = createBasicBlock("for.end"); | 
|  | 414 |  | 
|  | 415 | EmitBlock(CondBlock); | 
|  | 416 |  | 
|  | 417 | llvm::BasicBlock *ForBody = createBasicBlock("for.body"); | 
|  | 418 | // Generate: if (loop-index < number-of-elements fall to the loop body, | 
|  | 419 | // otherwise, go to the block after the for-loop. | 
| Anders Carlsson | 43db20e | 2010-05-01 17:02:18 +0000 | [diff] [blame] | 420 | uint64_t NumElements = getContext().getConstantArrayElementCount(Array); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 421 | llvm::Value * NumElementsPtr = | 
|  | 422 | llvm::ConstantInt::get(llvm::Type::getInt64Ty(VMContext), NumElements); | 
|  | 423 | llvm::Value *Counter = Builder.CreateLoad(IndexPtr); | 
|  | 424 | llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElementsPtr, | 
|  | 425 | "isless"); | 
|  | 426 | // If the condition is true, execute the body. | 
|  | 427 | Builder.CreateCondBr(IsLess, ForBody, AfterFor); | 
|  | 428 |  | 
|  | 429 | EmitBlock(ForBody); | 
|  | 430 | llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc"); | 
|  | 431 | // Inside the loop body, emit the constructor call on the array element. | 
|  | 432 | Counter = Builder.CreateLoad(IndexPtr); | 
| Anders Carlsson | f500de5 | 2010-04-24 22:43:39 +0000 | [diff] [blame] | 433 | Src = Builder.CreateInBoundsGEP(Src, Counter, "srcaddress"); | 
| Anders Carlsson | 8e142cc | 2010-04-25 00:52:09 +0000 | [diff] [blame] | 434 | Dest = Builder.CreateInBoundsGEP(Dest, Counter, "destaddress"); | 
| Anders Carlsson | 43db20e | 2010-05-01 17:02:18 +0000 | [diff] [blame] | 435 | EmitClassMemberwiseCopy(Dest, Src, ClassDecl); | 
|  | 436 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 437 | EmitBlock(ContinueBlock); | 
|  | 438 |  | 
|  | 439 | // Emit the increment of the loop counter. | 
|  | 440 | llvm::Value *NextVal = llvm::ConstantInt::get(Counter->getType(), 1); | 
|  | 441 | Counter = Builder.CreateLoad(IndexPtr); | 
|  | 442 | NextVal = Builder.CreateAdd(Counter, NextVal, "inc"); | 
|  | 443 | Builder.CreateStore(NextVal, IndexPtr); | 
|  | 444 |  | 
|  | 445 | // Finally, branch back up to the condition for the next iteration. | 
|  | 446 | EmitBranch(CondBlock); | 
|  | 447 |  | 
|  | 448 | // Emit the fall-through block. | 
|  | 449 | EmitBlock(AfterFor, true); | 
|  | 450 | } | 
|  | 451 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 452 | /// GetVTTParameter - Return the VTT parameter that should be passed to a | 
|  | 453 | /// base constructor/destructor with virtual bases. | 
|  | 454 | static llvm::Value *GetVTTParameter(CodeGenFunction &CGF, GlobalDecl GD) { | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 455 | if (!CodeGenVTables::needsVTTParameter(GD)) { | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 456 | // This constructor/destructor does not need a VTT parameter. | 
|  | 457 | return 0; | 
|  | 458 | } | 
|  | 459 |  | 
|  | 460 | const CXXRecordDecl *RD = cast<CXXMethodDecl>(CGF.CurFuncDecl)->getParent(); | 
|  | 461 | const CXXRecordDecl *Base = cast<CXXMethodDecl>(GD.getDecl())->getParent(); | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 462 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 463 | llvm::Value *VTT; | 
|  | 464 |  | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 465 | uint64_t SubVTTIndex; | 
|  | 466 |  | 
|  | 467 | // If the record matches the base, this is the complete ctor/dtor | 
|  | 468 | // variant calling the base variant in a class with virtual bases. | 
|  | 469 | if (RD == Base) { | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 470 | assert(!CodeGenVTables::needsVTTParameter(CGF.CurGD) && | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 471 | "doing no-op VTT offset in base dtor/ctor?"); | 
|  | 472 | SubVTTIndex = 0; | 
|  | 473 | } else { | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 474 | SubVTTIndex = CGF.CGM.getVTables().getSubVTTIndex(RD, Base); | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 475 | assert(SubVTTIndex != 0 && "Sub-VTT index must be greater than zero!"); | 
|  | 476 | } | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 477 |  | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 478 | if (CodeGenVTables::needsVTTParameter(CGF.CurGD)) { | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 479 | // A VTT parameter was passed to the constructor, use it. | 
|  | 480 | VTT = CGF.LoadCXXVTT(); | 
|  | 481 | VTT = CGF.Builder.CreateConstInBoundsGEP1_64(VTT, SubVTTIndex); | 
|  | 482 | } else { | 
|  | 483 | // We're the complete constructor, so get the VTT by name. | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 484 | VTT = CGF.CGM.getVTables().getVTT(RD); | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 485 | VTT = CGF.Builder.CreateConstInBoundsGEP2_64(VTT, 0, SubVTTIndex); | 
|  | 486 | } | 
|  | 487 |  | 
|  | 488 | return VTT; | 
|  | 489 | } | 
|  | 490 |  | 
|  | 491 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 492 | /// EmitClassMemberwiseCopy - This routine generates code to copy a class | 
| Anders Carlsson | 8e142cc | 2010-04-25 00:52:09 +0000 | [diff] [blame] | 493 | /// object from SrcValue to DestValue. Copying can be either a bitwise copy | 
|  | 494 | /// or via a copy constructor call. | 
|  | 495 | void CodeGenFunction::EmitClassMemberwiseCopy( | 
|  | 496 | llvm::Value *Dest, llvm::Value *Src, | 
| Anders Carlsson | f62756f | 2010-05-01 16:54:05 +0000 | [diff] [blame] | 497 | const CXXRecordDecl *ClassDecl) { | 
|  | 498 | if (ClassDecl->hasTrivialCopyConstructor()) { | 
|  | 499 | EmitAggregateCopy(Dest, Src, getContext().getTagDeclType(ClassDecl)); | 
| Anders Carlsson | 6444c41 | 2010-04-24 22:36:50 +0000 | [diff] [blame] | 500 | return; | 
|  | 501 | } | 
|  | 502 |  | 
| Anders Carlsson | f62756f | 2010-05-01 16:54:05 +0000 | [diff] [blame] | 503 | CXXConstructorDecl *CopyCtor = ClassDecl->getCopyConstructor(getContext(), 0); | 
|  | 504 | assert(CopyCtor && "Did not have copy ctor!"); | 
| Anders Carlsson | 8e142cc | 2010-04-25 00:52:09 +0000 | [diff] [blame] | 505 |  | 
| Anders Carlsson | bfe7e91 | 2010-05-01 17:07:40 +0000 | [diff] [blame] | 506 | EmitCopyCtorCall(*this, CopyCtor, Dest, Src); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 507 | } | 
|  | 508 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 509 | /// SynthesizeCXXCopyConstructor - This routine implicitly defines body of a | 
|  | 510 | /// copy constructor, in accordance with section 12.8 (p7 and p8) of C++03 | 
|  | 511 | /// The implicitly-defined copy constructor for class X performs a memberwise | 
|  | 512 | /// copy of its subobjects. The order of copying is the same as the order of | 
|  | 513 | /// initialization of bases and members in a user-defined constructor | 
|  | 514 | /// Each subobject is copied in the manner appropriate to its type: | 
|  | 515 | ///  if the subobject is of class type, the copy constructor for the class is | 
|  | 516 | ///  used; | 
|  | 517 | ///  if the subobject is an array, each element is copied, in the manner | 
|  | 518 | ///  appropriate to the element type; | 
|  | 519 | ///  if the subobject is of scalar type, the built-in assignment operator is | 
|  | 520 | ///  used. | 
|  | 521 | /// Virtual base class subobjects shall be copied only once by the | 
|  | 522 | /// implicitly-defined copy constructor | 
|  | 523 |  | 
|  | 524 | void | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 525 | CodeGenFunction::SynthesizeCXXCopyConstructor(const FunctionArgList &Args) { | 
|  | 526 | const CXXConstructorDecl *Ctor = cast<CXXConstructorDecl>(CurGD.getDecl()); | 
| John McCall | c743571 | 2010-04-30 05:56:45 +0000 | [diff] [blame] | 527 | CXXCtorType CtorType = CurGD.getCtorType(); | 
|  | 528 | (void) CtorType; | 
|  | 529 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 530 | const CXXRecordDecl *ClassDecl = Ctor->getParent(); | 
|  | 531 | assert(!ClassDecl->hasUserDeclaredCopyConstructor() && | 
|  | 532 | "SynthesizeCXXCopyConstructor - copy constructor has definition already"); | 
|  | 533 | assert(!Ctor->isTrivial() && "shouldn't need to generate trivial ctor"); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 534 |  | 
| Anders Carlsson | 9675466 | 2010-04-25 01:03:12 +0000 | [diff] [blame] | 535 | llvm::Value *ThisPtr = LoadCXXThis(); | 
| John McCall | c743571 | 2010-04-30 05:56:45 +0000 | [diff] [blame] | 536 |  | 
|  | 537 | // Find the source pointer. | 
|  | 538 | unsigned SrcArgIndex = Args.size() - 1; | 
|  | 539 | assert(CtorType == Ctor_Base || SrcArgIndex == 1); | 
|  | 540 | assert(CtorType != Ctor_Base || | 
|  | 541 | (ClassDecl->getNumVBases() != 0 && SrcArgIndex == 2) || | 
|  | 542 | SrcArgIndex == 1); | 
|  | 543 |  | 
|  | 544 | llvm::Value *SrcPtr = | 
|  | 545 | Builder.CreateLoad(GetAddrOfLocalVar(Args[SrcArgIndex].first)); | 
| Anders Carlsson | 8e142cc | 2010-04-25 00:52:09 +0000 | [diff] [blame] | 546 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 547 | for (CXXRecordDecl::field_iterator I = ClassDecl->field_begin(), | 
|  | 548 | E = ClassDecl->field_end(); I != E; ++I) { | 
|  | 549 | const FieldDecl *Field = *I; | 
|  | 550 |  | 
|  | 551 | QualType FieldType = getContext().getCanonicalType(Field->getType()); | 
|  | 552 | const ConstantArrayType *Array = | 
|  | 553 | getContext().getAsConstantArrayType(FieldType); | 
|  | 554 | if (Array) | 
|  | 555 | FieldType = getContext().getBaseElementType(FieldType); | 
|  | 556 |  | 
|  | 557 | if (const RecordType *FieldClassType = FieldType->getAs<RecordType>()) { | 
|  | 558 | CXXRecordDecl *FieldClassDecl | 
|  | 559 | = cast<CXXRecordDecl>(FieldClassType->getDecl()); | 
| Anders Carlsson | 9675466 | 2010-04-25 01:03:12 +0000 | [diff] [blame] | 560 | LValue LHS = EmitLValueForField(ThisPtr, Field, 0); | 
|  | 561 | LValue RHS = EmitLValueForField(SrcPtr, Field, 0); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 562 | if (Array) { | 
| Anders Carlsson | 9675466 | 2010-04-25 01:03:12 +0000 | [diff] [blame] | 563 | const llvm::Type *BasePtr = ConvertType(FieldType)->getPointerTo(); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 564 | llvm::Value *DestBaseAddrPtr = | 
|  | 565 | Builder.CreateBitCast(LHS.getAddress(), BasePtr); | 
|  | 566 | llvm::Value *SrcBaseAddrPtr = | 
|  | 567 | Builder.CreateBitCast(RHS.getAddress(), BasePtr); | 
|  | 568 | EmitClassAggrMemberwiseCopy(DestBaseAddrPtr, SrcBaseAddrPtr, Array, | 
| Anders Carlsson | 43db20e | 2010-05-01 17:02:18 +0000 | [diff] [blame] | 569 | FieldClassDecl); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 570 | } | 
|  | 571 | else | 
|  | 572 | EmitClassMemberwiseCopy(LHS.getAddress(), RHS.getAddress(), | 
| Anders Carlsson | f62756f | 2010-05-01 16:54:05 +0000 | [diff] [blame] | 573 | FieldClassDecl); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 574 | continue; | 
|  | 575 | } | 
|  | 576 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 577 | // Do a built-in assignment of scalar data members. | 
| Anders Carlsson | 9675466 | 2010-04-25 01:03:12 +0000 | [diff] [blame] | 578 | LValue LHS = EmitLValueForFieldInitialization(ThisPtr, Field, 0); | 
|  | 579 | LValue RHS = EmitLValueForFieldInitialization(SrcPtr, Field, 0); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 580 |  | 
|  | 581 | if (!hasAggregateLLVMType(Field->getType())) { | 
|  | 582 | RValue RVRHS = EmitLoadOfLValue(RHS, Field->getType()); | 
|  | 583 | EmitStoreThroughLValue(RVRHS, LHS, Field->getType()); | 
|  | 584 | } else if (Field->getType()->isAnyComplexType()) { | 
|  | 585 | ComplexPairTy Pair = LoadComplexFromAddr(RHS.getAddress(), | 
|  | 586 | RHS.isVolatileQualified()); | 
|  | 587 | StoreComplexToAddr(Pair, LHS.getAddress(), LHS.isVolatileQualified()); | 
|  | 588 | } else { | 
|  | 589 | EmitAggregateCopy(LHS.getAddress(), RHS.getAddress(), Field->getType()); | 
|  | 590 | } | 
|  | 591 | } | 
|  | 592 |  | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 593 | InitializeVTablePointers(ClassDecl); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 594 | } | 
|  | 595 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 596 | static void EmitBaseInitializer(CodeGenFunction &CGF, | 
|  | 597 | const CXXRecordDecl *ClassDecl, | 
|  | 598 | CXXBaseOrMemberInitializer *BaseInit, | 
|  | 599 | CXXCtorType CtorType) { | 
|  | 600 | assert(BaseInit->isBaseInitializer() && | 
|  | 601 | "Must have base initializer!"); | 
|  | 602 |  | 
|  | 603 | llvm::Value *ThisPtr = CGF.LoadCXXThis(); | 
|  | 604 |  | 
|  | 605 | const Type *BaseType = BaseInit->getBaseClass(); | 
|  | 606 | CXXRecordDecl *BaseClassDecl = | 
|  | 607 | cast<CXXRecordDecl>(BaseType->getAs<RecordType>()->getDecl()); | 
|  | 608 |  | 
| Anders Carlsson | 80638c5 | 2010-04-12 00:51:03 +0000 | [diff] [blame] | 609 | bool isBaseVirtual = BaseInit->isBaseVirtual(); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 610 |  | 
|  | 611 | // The base constructor doesn't construct virtual bases. | 
|  | 612 | if (CtorType == Ctor_Base && isBaseVirtual) | 
|  | 613 | return; | 
|  | 614 |  | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 615 | // We can pretend to be a complete class because it only matters for | 
|  | 616 | // virtual bases, and we only do virtual bases for complete ctors. | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 617 | llvm::Value *V = | 
|  | 618 | CGF.GetAddressOfDirectBaseInCompleteClass(ThisPtr, ClassDecl, | 
|  | 619 | BaseClassDecl, | 
|  | 620 | BaseInit->isBaseVirtual()); | 
| John McCall | bff225e | 2010-02-16 04:15:37 +0000 | [diff] [blame] | 621 |  | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 622 | CGF.EmitAggExpr(BaseInit->getInit(), V, false, false, true); | 
| Anders Carlsson | 594d5e8 | 2010-02-06 20:00:21 +0000 | [diff] [blame] | 623 |  | 
|  | 624 | if (CGF.Exceptions && !BaseClassDecl->hasTrivialDestructor()) { | 
|  | 625 | // FIXME: Is this OK for C++0x delegating constructors? | 
|  | 626 | CodeGenFunction::EHCleanupBlock Cleanup(CGF); | 
|  | 627 |  | 
| Anders Carlsson | 594d5e8 | 2010-02-06 20:00:21 +0000 | [diff] [blame] | 628 | CXXDestructorDecl *DD = BaseClassDecl->getDestructor(CGF.getContext()); | 
|  | 629 | CGF.EmitCXXDestructorCall(DD, Dtor_Base, V); | 
|  | 630 | } | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 631 | } | 
|  | 632 |  | 
|  | 633 | static void EmitMemberInitializer(CodeGenFunction &CGF, | 
|  | 634 | const CXXRecordDecl *ClassDecl, | 
|  | 635 | CXXBaseOrMemberInitializer *MemberInit) { | 
|  | 636 | assert(MemberInit->isMemberInitializer() && | 
|  | 637 | "Must have member initializer!"); | 
|  | 638 |  | 
|  | 639 | // non-static data member initializers. | 
|  | 640 | FieldDecl *Field = MemberInit->getMember(); | 
|  | 641 | QualType FieldType = CGF.getContext().getCanonicalType(Field->getType()); | 
|  | 642 |  | 
|  | 643 | llvm::Value *ThisPtr = CGF.LoadCXXThis(); | 
| Anders Carlsson | 06a2970 | 2010-01-29 05:24:29 +0000 | [diff] [blame] | 644 | LValue LHS = CGF.EmitLValueForFieldInitialization(ThisPtr, Field, 0); | 
|  | 645 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 646 | // If we are initializing an anonymous union field, drill down to the field. | 
|  | 647 | if (MemberInit->getAnonUnionMember()) { | 
|  | 648 | Field = MemberInit->getAnonUnionMember(); | 
| Anders Carlsson | e6d2a53 | 2010-01-29 05:05:36 +0000 | [diff] [blame] | 649 | LHS = CGF.EmitLValueForField(LHS.getAddress(), Field, 0); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 650 | FieldType = Field->getType(); | 
|  | 651 | } | 
|  | 652 |  | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 653 | // FIXME: If there's no initializer and the CXXBaseOrMemberInitializer | 
|  | 654 | // was implicitly generated, we shouldn't be zeroing memory. | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 655 | RValue RHS; | 
|  | 656 | if (FieldType->isReferenceType()) { | 
| Anders Carlsson | a64a869 | 2010-02-03 16:38:03 +0000 | [diff] [blame] | 657 | RHS = CGF.EmitReferenceBindingToExpr(MemberInit->getInit(), | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 658 | /*IsInitializer=*/true); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 659 | CGF.EmitStoreThroughLValue(RHS, LHS, FieldType); | 
| Eli Friedman | 3bb9412 | 2010-01-31 19:07:50 +0000 | [diff] [blame] | 660 | } else if (FieldType->isArrayType() && !MemberInit->getInit()) { | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 661 | CGF.EmitMemSetToZero(LHS.getAddress(), Field->getType()); | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 662 | } else if (!CGF.hasAggregateLLVMType(Field->getType())) { | 
|  | 663 | RHS = RValue::get(CGF.EmitScalarExpr(MemberInit->getInit(), true)); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 664 | CGF.EmitStoreThroughLValue(RHS, LHS, FieldType); | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 665 | } else if (MemberInit->getInit()->getType()->isAnyComplexType()) { | 
|  | 666 | CGF.EmitComplexExprIntoAddr(MemberInit->getInit(), LHS.getAddress(), | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 667 | LHS.isVolatileQualified()); | 
|  | 668 | } else { | 
| Douglas Gregor | 9db7dbb | 2010-01-31 09:12:51 +0000 | [diff] [blame] | 669 | CGF.EmitAggExpr(MemberInit->getInit(), LHS.getAddress(), | 
|  | 670 | LHS.isVolatileQualified(), false, true); | 
| Anders Carlsson | 9405dcd | 2010-02-06 19:50:17 +0000 | [diff] [blame] | 671 |  | 
|  | 672 | if (!CGF.Exceptions) | 
|  | 673 | return; | 
|  | 674 |  | 
|  | 675 | const RecordType *RT = FieldType->getAs<RecordType>(); | 
|  | 676 | if (!RT) | 
|  | 677 | return; | 
|  | 678 |  | 
|  | 679 | CXXRecordDecl *RD = cast<CXXRecordDecl>(RT->getDecl()); | 
|  | 680 | if (!RD->hasTrivialDestructor()) { | 
|  | 681 | // FIXME: Is this OK for C++0x delegating constructors? | 
|  | 682 | CodeGenFunction::EHCleanupBlock Cleanup(CGF); | 
|  | 683 |  | 
|  | 684 | llvm::Value *ThisPtr = CGF.LoadCXXThis(); | 
|  | 685 | LValue LHS = CGF.EmitLValueForField(ThisPtr, Field, 0); | 
|  | 686 |  | 
|  | 687 | CXXDestructorDecl *DD = RD->getDestructor(CGF.getContext()); | 
|  | 688 | CGF.EmitCXXDestructorCall(DD, Dtor_Complete, LHS.getAddress()); | 
|  | 689 | } | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 690 | } | 
|  | 691 | } | 
|  | 692 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 693 | /// Checks whether the given constructor is a valid subject for the | 
|  | 694 | /// complete-to-base constructor delegation optimization, i.e. | 
|  | 695 | /// emitting the complete constructor as a simple call to the base | 
|  | 696 | /// constructor. | 
|  | 697 | static bool IsConstructorDelegationValid(const CXXConstructorDecl *Ctor) { | 
|  | 698 |  | 
|  | 699 | // Currently we disable the optimization for classes with virtual | 
|  | 700 | // bases because (1) the addresses of parameter variables need to be | 
|  | 701 | // consistent across all initializers but (2) the delegate function | 
|  | 702 | // call necessarily creates a second copy of the parameter variable. | 
|  | 703 | // | 
|  | 704 | // The limiting example (purely theoretical AFAIK): | 
|  | 705 | //   struct A { A(int &c) { c++; } }; | 
|  | 706 | //   struct B : virtual A { | 
|  | 707 | //     B(int count) : A(count) { printf("%d\n", count); } | 
|  | 708 | //   }; | 
|  | 709 | // ...although even this example could in principle be emitted as a | 
|  | 710 | // delegation since the address of the parameter doesn't escape. | 
|  | 711 | if (Ctor->getParent()->getNumVBases()) { | 
|  | 712 | // TODO: white-list trivial vbase initializers.  This case wouldn't | 
|  | 713 | // be subject to the restrictions below. | 
|  | 714 |  | 
|  | 715 | // TODO: white-list cases where: | 
|  | 716 | //  - there are no non-reference parameters to the constructor | 
|  | 717 | //  - the initializers don't access any non-reference parameters | 
|  | 718 | //  - the initializers don't take the address of non-reference | 
|  | 719 | //    parameters | 
|  | 720 | //  - etc. | 
|  | 721 | // If we ever add any of the above cases, remember that: | 
|  | 722 | //  - function-try-blocks will always blacklist this optimization | 
|  | 723 | //  - we need to perform the constructor prologue and cleanup in | 
|  | 724 | //    EmitConstructorBody. | 
|  | 725 |  | 
|  | 726 | return false; | 
|  | 727 | } | 
|  | 728 |  | 
|  | 729 | // We also disable the optimization for variadic functions because | 
|  | 730 | // it's impossible to "re-pass" varargs. | 
|  | 731 | if (Ctor->getType()->getAs<FunctionProtoType>()->isVariadic()) | 
|  | 732 | return false; | 
|  | 733 |  | 
|  | 734 | return true; | 
|  | 735 | } | 
|  | 736 |  | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 737 | /// EmitConstructorBody - Emits the body of the current constructor. | 
|  | 738 | void CodeGenFunction::EmitConstructorBody(FunctionArgList &Args) { | 
|  | 739 | const CXXConstructorDecl *Ctor = cast<CXXConstructorDecl>(CurGD.getDecl()); | 
|  | 740 | CXXCtorType CtorType = CurGD.getCtorType(); | 
|  | 741 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 742 | // Before we go any further, try the complete->base constructor | 
|  | 743 | // delegation optimization. | 
|  | 744 | if (CtorType == Ctor_Complete && IsConstructorDelegationValid(Ctor)) { | 
|  | 745 | EmitDelegateCXXConstructorCall(Ctor, Ctor_Base, Args); | 
|  | 746 | return; | 
|  | 747 | } | 
|  | 748 |  | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 749 | Stmt *Body = Ctor->getBody(); | 
|  | 750 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 751 | // Enter the function-try-block before the constructor prologue if | 
|  | 752 | // applicable. | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 753 | CXXTryStmtInfo TryInfo; | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 754 | bool IsTryBody = (Body && isa<CXXTryStmt>(Body)); | 
|  | 755 |  | 
|  | 756 | if (IsTryBody) | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 757 | TryInfo = EnterCXXTryStmt(*cast<CXXTryStmt>(Body)); | 
|  | 758 |  | 
|  | 759 | unsigned CleanupStackSize = CleanupEntries.size(); | 
|  | 760 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 761 | // Emit the constructor prologue, i.e. the base and member | 
|  | 762 | // initializers. | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 763 | EmitCtorPrologue(Ctor, CtorType); | 
|  | 764 |  | 
|  | 765 | // Emit the body of the statement. | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 766 | if (IsTryBody) | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 767 | EmitStmt(cast<CXXTryStmt>(Body)->getTryBlock()); | 
|  | 768 | else if (Body) | 
|  | 769 | EmitStmt(Body); | 
|  | 770 | else { | 
|  | 771 | assert(Ctor->isImplicit() && "bodyless ctor not implicit"); | 
|  | 772 | if (!Ctor->isDefaultConstructor()) { | 
|  | 773 | assert(Ctor->isCopyConstructor()); | 
|  | 774 | SynthesizeCXXCopyConstructor(Args); | 
|  | 775 | } | 
|  | 776 | } | 
|  | 777 |  | 
|  | 778 | // Emit any cleanup blocks associated with the member or base | 
|  | 779 | // initializers, which includes (along the exceptional path) the | 
|  | 780 | // destructors for those members and bases that were fully | 
|  | 781 | // constructed. | 
|  | 782 | EmitCleanupBlocks(CleanupStackSize); | 
|  | 783 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 784 | if (IsTryBody) | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 785 | ExitCXXTryStmt(*cast<CXXTryStmt>(Body), TryInfo); | 
|  | 786 | } | 
|  | 787 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 788 | /// EmitCtorPrologue - This routine generates necessary code to initialize | 
|  | 789 | /// base classes and non-static data members belonging to this constructor. | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 790 | void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD, | 
|  | 791 | CXXCtorType CtorType) { | 
|  | 792 | const CXXRecordDecl *ClassDecl = CD->getParent(); | 
| Anders Carlsson | a78fa2c | 2010-02-02 19:58:43 +0000 | [diff] [blame] | 793 |  | 
|  | 794 | llvm::SmallVector<CXXBaseOrMemberInitializer *, 8> MemberInitializers; | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 795 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 796 | for (CXXConstructorDecl::init_const_iterator B = CD->init_begin(), | 
|  | 797 | E = CD->init_end(); | 
|  | 798 | B != E; ++B) { | 
|  | 799 | CXXBaseOrMemberInitializer *Member = (*B); | 
|  | 800 |  | 
|  | 801 | assert(LiveTemporaries.empty() && | 
|  | 802 | "Should not have any live temporaries at initializer start!"); | 
|  | 803 |  | 
|  | 804 | if (Member->isBaseInitializer()) | 
|  | 805 | EmitBaseInitializer(*this, ClassDecl, Member, CtorType); | 
|  | 806 | else | 
| Anders Carlsson | a78fa2c | 2010-02-02 19:58:43 +0000 | [diff] [blame] | 807 | MemberInitializers.push_back(Member); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 808 | } | 
|  | 809 |  | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 810 | InitializeVTablePointers(ClassDecl); | 
| Anders Carlsson | a78fa2c | 2010-02-02 19:58:43 +0000 | [diff] [blame] | 811 |  | 
|  | 812 | for (unsigned I = 0, E = MemberInitializers.size(); I != E; ++I) { | 
|  | 813 | assert(LiveTemporaries.empty() && | 
|  | 814 | "Should not have any live temporaries at initializer start!"); | 
|  | 815 |  | 
|  | 816 | EmitMemberInitializer(*this, ClassDecl, MemberInitializers[I]); | 
|  | 817 | } | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 818 | } | 
|  | 819 |  | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 820 | /// EmitDestructorBody - Emits the body of the current destructor. | 
|  | 821 | void CodeGenFunction::EmitDestructorBody(FunctionArgList &Args) { | 
|  | 822 | const CXXDestructorDecl *Dtor = cast<CXXDestructorDecl>(CurGD.getDecl()); | 
|  | 823 | CXXDtorType DtorType = CurGD.getDtorType(); | 
|  | 824 |  | 
|  | 825 | Stmt *Body = Dtor->getBody(); | 
|  | 826 |  | 
|  | 827 | // If the body is a function-try-block, enter the try before | 
|  | 828 | // anything else --- unless we're in a deleting destructor, in which | 
|  | 829 | // case we're just going to call the complete destructor and then | 
|  | 830 | // call operator delete() on the way out. | 
|  | 831 | CXXTryStmtInfo TryInfo; | 
|  | 832 | bool isTryBody = (DtorType != Dtor_Deleting && | 
|  | 833 | Body && isa<CXXTryStmt>(Body)); | 
|  | 834 | if (isTryBody) | 
|  | 835 | TryInfo = EnterCXXTryStmt(*cast<CXXTryStmt>(Body)); | 
|  | 836 |  | 
|  | 837 | llvm::BasicBlock *DtorEpilogue = createBasicBlock("dtor.epilogue"); | 
|  | 838 | PushCleanupBlock(DtorEpilogue); | 
|  | 839 |  | 
|  | 840 | bool SkipBody = false; // should get jump-threaded | 
|  | 841 |  | 
|  | 842 | // If this is the deleting variant, just invoke the complete | 
|  | 843 | // variant, then call the appropriate operator delete() on the way | 
|  | 844 | // out. | 
|  | 845 | if (DtorType == Dtor_Deleting) { | 
|  | 846 | EmitCXXDestructorCall(Dtor, Dtor_Complete, LoadCXXThis()); | 
|  | 847 | SkipBody = true; | 
|  | 848 |  | 
|  | 849 | // If this is the complete variant, just invoke the base variant; | 
|  | 850 | // the epilogue will destruct the virtual bases.  But we can't do | 
|  | 851 | // this optimization if the body is a function-try-block, because | 
|  | 852 | // we'd introduce *two* handler blocks. | 
|  | 853 | } else if (!isTryBody && DtorType == Dtor_Complete) { | 
|  | 854 | EmitCXXDestructorCall(Dtor, Dtor_Base, LoadCXXThis()); | 
|  | 855 | SkipBody = true; | 
|  | 856 |  | 
|  | 857 | // Otherwise, we're in the base variant, so we need to ensure the | 
|  | 858 | // vtable ptrs are right before emitting the body. | 
|  | 859 | } else { | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 860 | InitializeVTablePointers(Dtor->getParent()); | 
| John McCall | 9fc6a77 | 2010-02-19 09:25:03 +0000 | [diff] [blame] | 861 | } | 
|  | 862 |  | 
|  | 863 | // Emit the body of the statement. | 
|  | 864 | if (SkipBody) | 
|  | 865 | (void) 0; | 
|  | 866 | else if (isTryBody) | 
|  | 867 | EmitStmt(cast<CXXTryStmt>(Body)->getTryBlock()); | 
|  | 868 | else if (Body) | 
|  | 869 | EmitStmt(Body); | 
|  | 870 | else { | 
|  | 871 | assert(Dtor->isImplicit() && "bodyless dtor not implicit"); | 
|  | 872 | // nothing to do besides what's in the epilogue | 
|  | 873 | } | 
|  | 874 |  | 
|  | 875 | // Jump to the cleanup block. | 
|  | 876 | CleanupBlockInfo Info = PopCleanupBlock(); | 
|  | 877 | assert(Info.CleanupBlock == DtorEpilogue && "Block mismatch!"); | 
|  | 878 | EmitBlock(DtorEpilogue); | 
|  | 879 |  | 
|  | 880 | // Emit the destructor epilogue now.  If this is a complete | 
|  | 881 | // destructor with a function-try-block, perform the base epilogue | 
|  | 882 | // as well. | 
|  | 883 | if (isTryBody && DtorType == Dtor_Complete) | 
|  | 884 | EmitDtorEpilogue(Dtor, Dtor_Base); | 
|  | 885 | EmitDtorEpilogue(Dtor, DtorType); | 
|  | 886 |  | 
|  | 887 | // Link up the cleanup information. | 
|  | 888 | if (Info.SwitchBlock) | 
|  | 889 | EmitBlock(Info.SwitchBlock); | 
|  | 890 | if (Info.EndBlock) | 
|  | 891 | EmitBlock(Info.EndBlock); | 
|  | 892 |  | 
|  | 893 | // Exit the try if applicable. | 
|  | 894 | if (isTryBody) | 
|  | 895 | ExitCXXTryStmt(*cast<CXXTryStmt>(Body), TryInfo); | 
|  | 896 | } | 
|  | 897 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 898 | /// EmitDtorEpilogue - Emit all code that comes at the end of class's | 
|  | 899 | /// destructor. This is to call destructors on members and base classes | 
|  | 900 | /// in reverse order of their construction. | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 901 | void CodeGenFunction::EmitDtorEpilogue(const CXXDestructorDecl *DD, | 
|  | 902 | CXXDtorType DtorType) { | 
|  | 903 | assert(!DD->isTrivial() && | 
|  | 904 | "Should not emit dtor epilogue for trivial dtor!"); | 
|  | 905 |  | 
|  | 906 | const CXXRecordDecl *ClassDecl = DD->getParent(); | 
|  | 907 |  | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 908 | // In a deleting destructor, we've already called the complete | 
|  | 909 | // destructor as a subroutine, so we just have to delete the | 
|  | 910 | // appropriate value. | 
|  | 911 | if (DtorType == Dtor_Deleting) { | 
|  | 912 | assert(DD->getOperatorDelete() && | 
|  | 913 | "operator delete missing - EmitDtorEpilogue"); | 
|  | 914 | EmitDeleteCall(DD->getOperatorDelete(), LoadCXXThis(), | 
|  | 915 | getContext().getTagDeclType(ClassDecl)); | 
|  | 916 | return; | 
|  | 917 | } | 
|  | 918 |  | 
|  | 919 | // For complete destructors, we've already called the base | 
|  | 920 | // destructor (in GenerateBody), so we just need to destruct all the | 
|  | 921 | // virtual bases. | 
|  | 922 | if (DtorType == Dtor_Complete) { | 
|  | 923 | // Handle virtual bases. | 
|  | 924 | for (CXXRecordDecl::reverse_base_class_const_iterator I = | 
|  | 925 | ClassDecl->vbases_rbegin(), E = ClassDecl->vbases_rend(); | 
|  | 926 | I != E; ++I) { | 
|  | 927 | const CXXBaseSpecifier &Base = *I; | 
|  | 928 | CXXRecordDecl *BaseClassDecl | 
|  | 929 | = cast<CXXRecordDecl>(Base.getType()->getAs<RecordType>()->getDecl()); | 
|  | 930 |  | 
|  | 931 | // Ignore trivial destructors. | 
|  | 932 | if (BaseClassDecl->hasTrivialDestructor()) | 
|  | 933 | continue; | 
|  | 934 | const CXXDestructorDecl *D = BaseClassDecl->getDestructor(getContext()); | 
| Anders Carlsson | 8561a86 | 2010-04-24 23:01:49 +0000 | [diff] [blame] | 935 | llvm::Value *V = | 
|  | 936 | GetAddressOfDirectBaseInCompleteClass(LoadCXXThis(), | 
|  | 937 | ClassDecl, BaseClassDecl, | 
|  | 938 | /*BaseIsVirtual=*/true); | 
| John McCall | 3b47733 | 2010-02-18 19:59:28 +0000 | [diff] [blame] | 939 | EmitCXXDestructorCall(D, Dtor_Base, V); | 
|  | 940 | } | 
|  | 941 | return; | 
|  | 942 | } | 
|  | 943 |  | 
|  | 944 | assert(DtorType == Dtor_Base); | 
|  | 945 |  | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 946 | // Collect the fields. | 
|  | 947 | llvm::SmallVector<const FieldDecl *, 16> FieldDecls; | 
|  | 948 | for (CXXRecordDecl::field_iterator I = ClassDecl->field_begin(), | 
|  | 949 | E = ClassDecl->field_end(); I != E; ++I) { | 
|  | 950 | const FieldDecl *Field = *I; | 
|  | 951 |  | 
|  | 952 | QualType FieldType = getContext().getCanonicalType(Field->getType()); | 
|  | 953 | FieldType = getContext().getBaseElementType(FieldType); | 
|  | 954 |  | 
|  | 955 | const RecordType *RT = FieldType->getAs<RecordType>(); | 
|  | 956 | if (!RT) | 
|  | 957 | continue; | 
|  | 958 |  | 
|  | 959 | CXXRecordDecl *FieldClassDecl = cast<CXXRecordDecl>(RT->getDecl()); | 
|  | 960 | if (FieldClassDecl->hasTrivialDestructor()) | 
|  | 961 | continue; | 
|  | 962 |  | 
|  | 963 | FieldDecls.push_back(Field); | 
|  | 964 | } | 
|  | 965 |  | 
|  | 966 | // Now destroy the fields. | 
|  | 967 | for (size_t i = FieldDecls.size(); i > 0; --i) { | 
|  | 968 | const FieldDecl *Field = FieldDecls[i - 1]; | 
|  | 969 |  | 
|  | 970 | QualType FieldType = Field->getType(); | 
|  | 971 | const ConstantArrayType *Array = | 
|  | 972 | getContext().getAsConstantArrayType(FieldType); | 
|  | 973 | if (Array) | 
|  | 974 | FieldType = getContext().getBaseElementType(FieldType); | 
|  | 975 |  | 
|  | 976 | const RecordType *RT = FieldType->getAs<RecordType>(); | 
|  | 977 | CXXRecordDecl *FieldClassDecl = cast<CXXRecordDecl>(RT->getDecl()); | 
|  | 978 |  | 
|  | 979 | llvm::Value *ThisPtr = LoadCXXThis(); | 
|  | 980 |  | 
|  | 981 | LValue LHS = EmitLValueForField(ThisPtr, Field, | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 982 | // FIXME: Qualifiers? | 
|  | 983 | /*CVRQualifiers=*/0); | 
|  | 984 | if (Array) { | 
|  | 985 | const llvm::Type *BasePtr = ConvertType(FieldType); | 
|  | 986 | BasePtr = llvm::PointerType::getUnqual(BasePtr); | 
|  | 987 | llvm::Value *BaseAddrPtr = | 
|  | 988 | Builder.CreateBitCast(LHS.getAddress(), BasePtr); | 
|  | 989 | EmitCXXAggrDestructorCall(FieldClassDecl->getDestructor(getContext()), | 
|  | 990 | Array, BaseAddrPtr); | 
|  | 991 | } else | 
|  | 992 | EmitCXXDestructorCall(FieldClassDecl->getDestructor(getContext()), | 
|  | 993 | Dtor_Complete, LHS.getAddress()); | 
|  | 994 | } | 
|  | 995 |  | 
|  | 996 | // Destroy non-virtual bases. | 
|  | 997 | for (CXXRecordDecl::reverse_base_class_const_iterator I = | 
|  | 998 | ClassDecl->bases_rbegin(), E = ClassDecl->bases_rend(); I != E; ++I) { | 
|  | 999 | const CXXBaseSpecifier &Base = *I; | 
|  | 1000 |  | 
|  | 1001 | // Ignore virtual bases. | 
|  | 1002 | if (Base.isVirtual()) | 
|  | 1003 | continue; | 
|  | 1004 |  | 
|  | 1005 | CXXRecordDecl *BaseClassDecl | 
|  | 1006 | = cast<CXXRecordDecl>(Base.getType()->getAs<RecordType>()->getDecl()); | 
|  | 1007 |  | 
|  | 1008 | // Ignore trivial destructors. | 
|  | 1009 | if (BaseClassDecl->hasTrivialDestructor()) | 
|  | 1010 | continue; | 
|  | 1011 | const CXXDestructorDecl *D = BaseClassDecl->getDestructor(getContext()); | 
|  | 1012 |  | 
| Anders Carlsson | a88ad56 | 2010-04-24 21:51:08 +0000 | [diff] [blame] | 1013 | llvm::Value *V = OldGetAddressOfBaseClass(LoadCXXThis(), | 
|  | 1014 | ClassDecl, BaseClassDecl); | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 1015 | EmitCXXDestructorCall(D, Dtor_Base, V); | 
|  | 1016 | } | 
| Anders Carlsson | 607d037 | 2009-12-24 22:46:43 +0000 | [diff] [blame] | 1017 | } | 
|  | 1018 |  | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1019 | /// EmitCXXAggrConstructorCall - This routine essentially creates a (nested) | 
|  | 1020 | /// for-loop to call the default constructor on individual members of the | 
|  | 1021 | /// array. | 
|  | 1022 | /// 'D' is the default constructor for elements of the array, 'ArrayTy' is the | 
|  | 1023 | /// array type and 'ArrayPtr' points to the beginning fo the array. | 
|  | 1024 | /// It is assumed that all relevant checks have been made by the caller. | 
|  | 1025 | void | 
|  | 1026 | CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D, | 
|  | 1027 | const ConstantArrayType *ArrayTy, | 
|  | 1028 | llvm::Value *ArrayPtr, | 
|  | 1029 | CallExpr::const_arg_iterator ArgBeg, | 
|  | 1030 | CallExpr::const_arg_iterator ArgEnd) { | 
|  | 1031 |  | 
|  | 1032 | const llvm::Type *SizeTy = ConvertType(getContext().getSizeType()); | 
|  | 1033 | llvm::Value * NumElements = | 
|  | 1034 | llvm::ConstantInt::get(SizeTy, | 
|  | 1035 | getContext().getConstantArrayElementCount(ArrayTy)); | 
|  | 1036 |  | 
|  | 1037 | EmitCXXAggrConstructorCall(D, NumElements, ArrayPtr, ArgBeg, ArgEnd); | 
|  | 1038 | } | 
|  | 1039 |  | 
|  | 1040 | void | 
|  | 1041 | CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D, | 
|  | 1042 | llvm::Value *NumElements, | 
|  | 1043 | llvm::Value *ArrayPtr, | 
|  | 1044 | CallExpr::const_arg_iterator ArgBeg, | 
|  | 1045 | CallExpr::const_arg_iterator ArgEnd) { | 
|  | 1046 | const llvm::Type *SizeTy = ConvertType(getContext().getSizeType()); | 
|  | 1047 |  | 
|  | 1048 | // Create a temporary for the loop index and initialize it with 0. | 
|  | 1049 | llvm::Value *IndexPtr = CreateTempAlloca(SizeTy, "loop.index"); | 
|  | 1050 | llvm::Value *Zero = llvm::Constant::getNullValue(SizeTy); | 
|  | 1051 | Builder.CreateStore(Zero, IndexPtr); | 
|  | 1052 |  | 
|  | 1053 | // Start the loop with a block that tests the condition. | 
|  | 1054 | llvm::BasicBlock *CondBlock = createBasicBlock("for.cond"); | 
|  | 1055 | llvm::BasicBlock *AfterFor = createBasicBlock("for.end"); | 
|  | 1056 |  | 
|  | 1057 | EmitBlock(CondBlock); | 
|  | 1058 |  | 
|  | 1059 | llvm::BasicBlock *ForBody = createBasicBlock("for.body"); | 
|  | 1060 |  | 
|  | 1061 | // Generate: if (loop-index < number-of-elements fall to the loop body, | 
|  | 1062 | // otherwise, go to the block after the for-loop. | 
|  | 1063 | llvm::Value *Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1064 | llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElements, "isless"); | 
|  | 1065 | // If the condition is true, execute the body. | 
|  | 1066 | Builder.CreateCondBr(IsLess, ForBody, AfterFor); | 
|  | 1067 |  | 
|  | 1068 | EmitBlock(ForBody); | 
|  | 1069 |  | 
|  | 1070 | llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc"); | 
|  | 1071 | // Inside the loop body, emit the constructor call on the array element. | 
|  | 1072 | Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1073 | llvm::Value *Address = Builder.CreateInBoundsGEP(ArrayPtr, Counter, | 
|  | 1074 | "arrayidx"); | 
|  | 1075 |  | 
|  | 1076 | // C++ [class.temporary]p4: | 
|  | 1077 | // There are two contexts in which temporaries are destroyed at a different | 
|  | 1078 | // point than the end of the full-expression. The first context is when a | 
|  | 1079 | // default constructor is called to initialize an element of an array. | 
|  | 1080 | // If the constructor has one or more default arguments, the destruction of | 
|  | 1081 | // every temporary created in a default argument expression is sequenced | 
|  | 1082 | // before the construction of the next array element, if any. | 
|  | 1083 |  | 
|  | 1084 | // Keep track of the current number of live temporaries. | 
| Anders Carlsson | 44ec82b | 2010-03-30 03:14:41 +0000 | [diff] [blame] | 1085 | { | 
|  | 1086 | CXXTemporariesCleanupScope Scope(*this); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1087 |  | 
| Anders Carlsson | 24eb78e | 2010-05-02 23:01:10 +0000 | [diff] [blame^] | 1088 | EmitCXXConstructorCall(D, CXXConstructExpr::CK_Complete, Address, | 
|  | 1089 | ArgBeg, ArgEnd); | 
| Anders Carlsson | 44ec82b | 2010-03-30 03:14:41 +0000 | [diff] [blame] | 1090 | } | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1091 |  | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1092 | EmitBlock(ContinueBlock); | 
|  | 1093 |  | 
|  | 1094 | // Emit the increment of the loop counter. | 
|  | 1095 | llvm::Value *NextVal = llvm::ConstantInt::get(SizeTy, 1); | 
|  | 1096 | Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1097 | NextVal = Builder.CreateAdd(Counter, NextVal, "inc"); | 
|  | 1098 | Builder.CreateStore(NextVal, IndexPtr); | 
|  | 1099 |  | 
|  | 1100 | // Finally, branch back up to the condition for the next iteration. | 
|  | 1101 | EmitBranch(CondBlock); | 
|  | 1102 |  | 
|  | 1103 | // Emit the fall-through block. | 
|  | 1104 | EmitBlock(AfterFor, true); | 
|  | 1105 | } | 
|  | 1106 |  | 
|  | 1107 | /// EmitCXXAggrDestructorCall - calls the default destructor on array | 
|  | 1108 | /// elements in reverse order of construction. | 
|  | 1109 | void | 
|  | 1110 | CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D, | 
|  | 1111 | const ArrayType *Array, | 
|  | 1112 | llvm::Value *This) { | 
|  | 1113 | const ConstantArrayType *CA = dyn_cast<ConstantArrayType>(Array); | 
|  | 1114 | assert(CA && "Do we support VLA for destruction ?"); | 
|  | 1115 | uint64_t ElementCount = getContext().getConstantArrayElementCount(CA); | 
|  | 1116 |  | 
|  | 1117 | const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType()); | 
|  | 1118 | llvm::Value* ElementCountPtr = llvm::ConstantInt::get(SizeLTy, ElementCount); | 
|  | 1119 | EmitCXXAggrDestructorCall(D, ElementCountPtr, This); | 
|  | 1120 | } | 
|  | 1121 |  | 
|  | 1122 | /// EmitCXXAggrDestructorCall - calls the default destructor on array | 
|  | 1123 | /// elements in reverse order of construction. | 
|  | 1124 | void | 
|  | 1125 | CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D, | 
|  | 1126 | llvm::Value *UpperCount, | 
|  | 1127 | llvm::Value *This) { | 
|  | 1128 | const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType()); | 
|  | 1129 | llvm::Value *One = llvm::ConstantInt::get(SizeLTy, 1); | 
|  | 1130 |  | 
|  | 1131 | // Create a temporary for the loop index and initialize it with count of | 
|  | 1132 | // array elements. | 
|  | 1133 | llvm::Value *IndexPtr = CreateTempAlloca(SizeLTy, "loop.index"); | 
|  | 1134 |  | 
|  | 1135 | // Store the number of elements in the index pointer. | 
|  | 1136 | Builder.CreateStore(UpperCount, IndexPtr); | 
|  | 1137 |  | 
|  | 1138 | // Start the loop with a block that tests the condition. | 
|  | 1139 | llvm::BasicBlock *CondBlock = createBasicBlock("for.cond"); | 
|  | 1140 | llvm::BasicBlock *AfterFor = createBasicBlock("for.end"); | 
|  | 1141 |  | 
|  | 1142 | EmitBlock(CondBlock); | 
|  | 1143 |  | 
|  | 1144 | llvm::BasicBlock *ForBody = createBasicBlock("for.body"); | 
|  | 1145 |  | 
|  | 1146 | // Generate: if (loop-index != 0 fall to the loop body, | 
|  | 1147 | // otherwise, go to the block after the for-loop. | 
|  | 1148 | llvm::Value* zeroConstant = | 
|  | 1149 | llvm::Constant::getNullValue(SizeLTy); | 
|  | 1150 | llvm::Value *Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1151 | llvm::Value *IsNE = Builder.CreateICmpNE(Counter, zeroConstant, | 
|  | 1152 | "isne"); | 
|  | 1153 | // If the condition is true, execute the body. | 
|  | 1154 | Builder.CreateCondBr(IsNE, ForBody, AfterFor); | 
|  | 1155 |  | 
|  | 1156 | EmitBlock(ForBody); | 
|  | 1157 |  | 
|  | 1158 | llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc"); | 
|  | 1159 | // Inside the loop body, emit the constructor call on the array element. | 
|  | 1160 | Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1161 | Counter = Builder.CreateSub(Counter, One); | 
|  | 1162 | llvm::Value *Address = Builder.CreateInBoundsGEP(This, Counter, "arrayidx"); | 
|  | 1163 | EmitCXXDestructorCall(D, Dtor_Complete, Address); | 
|  | 1164 |  | 
|  | 1165 | EmitBlock(ContinueBlock); | 
|  | 1166 |  | 
|  | 1167 | // Emit the decrement of the loop counter. | 
|  | 1168 | Counter = Builder.CreateLoad(IndexPtr); | 
|  | 1169 | Counter = Builder.CreateSub(Counter, One, "dec"); | 
|  | 1170 | Builder.CreateStore(Counter, IndexPtr); | 
|  | 1171 |  | 
|  | 1172 | // Finally, branch back up to the condition for the next iteration. | 
|  | 1173 | EmitBranch(CondBlock); | 
|  | 1174 |  | 
|  | 1175 | // Emit the fall-through block. | 
|  | 1176 | EmitBlock(AfterFor, true); | 
|  | 1177 | } | 
|  | 1178 |  | 
|  | 1179 | /// GenerateCXXAggrDestructorHelper - Generates a helper function which when | 
|  | 1180 | /// invoked, calls the default destructor on array elements in reverse order of | 
|  | 1181 | /// construction. | 
|  | 1182 | llvm::Constant * | 
|  | 1183 | CodeGenFunction::GenerateCXXAggrDestructorHelper(const CXXDestructorDecl *D, | 
|  | 1184 | const ArrayType *Array, | 
|  | 1185 | llvm::Value *This) { | 
|  | 1186 | FunctionArgList Args; | 
|  | 1187 | ImplicitParamDecl *Dst = | 
|  | 1188 | ImplicitParamDecl::Create(getContext(), 0, | 
|  | 1189 | SourceLocation(), 0, | 
|  | 1190 | getContext().getPointerType(getContext().VoidTy)); | 
|  | 1191 | Args.push_back(std::make_pair(Dst, Dst->getType())); | 
|  | 1192 |  | 
|  | 1193 | llvm::SmallString<16> Name; | 
|  | 1194 | llvm::raw_svector_ostream(Name) << "__tcf_" << (++UniqueAggrDestructorCount); | 
|  | 1195 | QualType R = getContext().VoidTy; | 
| John McCall | 04a67a6 | 2010-02-05 21:31:56 +0000 | [diff] [blame] | 1196 | const CGFunctionInfo &FI | 
| Rafael Espindola | 264ba48 | 2010-03-30 20:24:48 +0000 | [diff] [blame] | 1197 | = CGM.getTypes().getFunctionInfo(R, Args, FunctionType::ExtInfo()); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1198 | const llvm::FunctionType *FTy = CGM.getTypes().GetFunctionType(FI, false); | 
|  | 1199 | llvm::Function *Fn = | 
|  | 1200 | llvm::Function::Create(FTy, llvm::GlobalValue::InternalLinkage, | 
|  | 1201 | Name.str(), | 
|  | 1202 | &CGM.getModule()); | 
|  | 1203 | IdentifierInfo *II = &CGM.getContext().Idents.get(Name.str()); | 
|  | 1204 | FunctionDecl *FD = FunctionDecl::Create(getContext(), | 
|  | 1205 | getContext().getTranslationUnitDecl(), | 
|  | 1206 | SourceLocation(), II, R, 0, | 
|  | 1207 | FunctionDecl::Static, | 
| Douglas Gregor | 16573fa | 2010-04-19 22:54:31 +0000 | [diff] [blame] | 1208 | FunctionDecl::None, | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1209 | false, true); | 
|  | 1210 | StartFunction(FD, R, Fn, Args, SourceLocation()); | 
|  | 1211 | QualType BaseElementTy = getContext().getBaseElementType(Array); | 
|  | 1212 | const llvm::Type *BasePtr = ConvertType(BaseElementTy); | 
|  | 1213 | BasePtr = llvm::PointerType::getUnqual(BasePtr); | 
|  | 1214 | llvm::Value *BaseAddrPtr = Builder.CreateBitCast(This, BasePtr); | 
|  | 1215 | EmitCXXAggrDestructorCall(D, Array, BaseAddrPtr); | 
|  | 1216 | FinishFunction(); | 
|  | 1217 | llvm::Type *Ptr8Ty = llvm::PointerType::get(llvm::Type::getInt8Ty(VMContext), | 
|  | 1218 | 0); | 
|  | 1219 | llvm::Constant *m = llvm::ConstantExpr::getBitCast(Fn, Ptr8Ty); | 
|  | 1220 | return m; | 
|  | 1221 | } | 
|  | 1222 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 1223 |  | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1224 | void | 
|  | 1225 | CodeGenFunction::EmitCXXConstructorCall(const CXXConstructorDecl *D, | 
| Anders Carlsson | 24eb78e | 2010-05-02 23:01:10 +0000 | [diff] [blame^] | 1226 | CXXConstructExpr::ConstructionKind Kind, | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1227 | llvm::Value *This, | 
|  | 1228 | CallExpr::const_arg_iterator ArgBeg, | 
|  | 1229 | CallExpr::const_arg_iterator ArgEnd) { | 
| Anders Carlsson | 24eb78e | 2010-05-02 23:01:10 +0000 | [diff] [blame^] | 1230 | CXXCtorType Type = | 
|  | 1231 | (Kind == CXXConstructExpr::CK_Complete) ? Ctor_Complete : Ctor_Base; | 
|  | 1232 |  | 
| John McCall | 8b6bbeb | 2010-02-06 00:25:16 +0000 | [diff] [blame] | 1233 | if (D->isTrivial()) { | 
|  | 1234 | if (ArgBeg == ArgEnd) { | 
|  | 1235 | // Trivial default constructor, no codegen required. | 
|  | 1236 | assert(D->isDefaultConstructor() && | 
|  | 1237 | "trivial 0-arg ctor not a default ctor"); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1238 | return; | 
|  | 1239 | } | 
| John McCall | 8b6bbeb | 2010-02-06 00:25:16 +0000 | [diff] [blame] | 1240 |  | 
|  | 1241 | assert(ArgBeg + 1 == ArgEnd && "unexpected argcount for trivial ctor"); | 
|  | 1242 | assert(D->isCopyConstructor() && "trivial 1-arg ctor not a copy ctor"); | 
|  | 1243 |  | 
| John McCall | 8b6bbeb | 2010-02-06 00:25:16 +0000 | [diff] [blame] | 1244 | const Expr *E = (*ArgBeg); | 
|  | 1245 | QualType Ty = E->getType(); | 
|  | 1246 | llvm::Value *Src = EmitLValue(E).getAddress(); | 
|  | 1247 | EmitAggregateCopy(This, Src, Ty); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1248 | return; | 
|  | 1249 | } | 
|  | 1250 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 1251 | llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(D, Type)); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1252 | llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(D, Type); | 
|  | 1253 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 1254 | EmitCXXMemberCall(D, Callee, ReturnValueSlot(), This, VTT, ArgBeg, ArgEnd); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1255 | } | 
|  | 1256 |  | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 1257 | void | 
|  | 1258 | CodeGenFunction::EmitDelegateCXXConstructorCall(const CXXConstructorDecl *Ctor, | 
|  | 1259 | CXXCtorType CtorType, | 
|  | 1260 | const FunctionArgList &Args) { | 
|  | 1261 | CallArgList DelegateArgs; | 
|  | 1262 |  | 
|  | 1263 | FunctionArgList::const_iterator I = Args.begin(), E = Args.end(); | 
|  | 1264 | assert(I != E && "no parameters to constructor"); | 
|  | 1265 |  | 
|  | 1266 | // this | 
|  | 1267 | DelegateArgs.push_back(std::make_pair(RValue::get(LoadCXXThis()), | 
|  | 1268 | I->second)); | 
|  | 1269 | ++I; | 
|  | 1270 |  | 
|  | 1271 | // vtt | 
|  | 1272 | if (llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(Ctor, CtorType))) { | 
|  | 1273 | QualType VoidPP = getContext().getPointerType(getContext().VoidPtrTy); | 
|  | 1274 | DelegateArgs.push_back(std::make_pair(RValue::get(VTT), VoidPP)); | 
|  | 1275 |  | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 1276 | if (CodeGenVTables::needsVTTParameter(CurGD)) { | 
| John McCall | c0bf462 | 2010-02-23 00:48:20 +0000 | [diff] [blame] | 1277 | assert(I != E && "cannot skip vtt parameter, already done with args"); | 
|  | 1278 | assert(I->second == VoidPP && "skipping parameter not of vtt type"); | 
|  | 1279 | ++I; | 
|  | 1280 | } | 
|  | 1281 | } | 
|  | 1282 |  | 
|  | 1283 | // Explicit arguments. | 
|  | 1284 | for (; I != E; ++I) { | 
|  | 1285 |  | 
|  | 1286 | const VarDecl *Param = I->first; | 
|  | 1287 | QualType ArgType = Param->getType(); // because we're passing it to itself | 
|  | 1288 |  | 
|  | 1289 | // StartFunction converted the ABI-lowered parameter(s) into a | 
|  | 1290 | // local alloca.  We need to turn that into an r-value suitable | 
|  | 1291 | // for EmitCall. | 
|  | 1292 | llvm::Value *Local = GetAddrOfLocalVar(Param); | 
|  | 1293 | RValue Arg; | 
|  | 1294 |  | 
|  | 1295 | // For the most part, we just need to load the alloca, except: | 
|  | 1296 | // 1) aggregate r-values are actually pointers to temporaries, and | 
|  | 1297 | // 2) references to aggregates are pointers directly to the aggregate. | 
|  | 1298 | // I don't know why references to non-aggregates are different here. | 
|  | 1299 | if (ArgType->isReferenceType()) { | 
|  | 1300 | const ReferenceType *RefType = ArgType->getAs<ReferenceType>(); | 
|  | 1301 | if (hasAggregateLLVMType(RefType->getPointeeType())) | 
|  | 1302 | Arg = RValue::getAggregate(Local); | 
|  | 1303 | else | 
|  | 1304 | // Locals which are references to scalars are represented | 
|  | 1305 | // with allocas holding the pointer. | 
|  | 1306 | Arg = RValue::get(Builder.CreateLoad(Local)); | 
|  | 1307 | } else { | 
|  | 1308 | if (hasAggregateLLVMType(ArgType)) | 
|  | 1309 | Arg = RValue::getAggregate(Local); | 
|  | 1310 | else | 
|  | 1311 | Arg = RValue::get(EmitLoadOfScalar(Local, false, ArgType)); | 
|  | 1312 | } | 
|  | 1313 |  | 
|  | 1314 | DelegateArgs.push_back(std::make_pair(Arg, ArgType)); | 
|  | 1315 | } | 
|  | 1316 |  | 
|  | 1317 | EmitCall(CGM.getTypes().getFunctionInfo(Ctor, CtorType), | 
|  | 1318 | CGM.GetAddrOfCXXConstructor(Ctor, CtorType), | 
|  | 1319 | ReturnValueSlot(), DelegateArgs, Ctor); | 
|  | 1320 | } | 
|  | 1321 |  | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1322 | void CodeGenFunction::EmitCXXDestructorCall(const CXXDestructorDecl *DD, | 
|  | 1323 | CXXDtorType Type, | 
|  | 1324 | llvm::Value *This) { | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 1325 | llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(DD, Type)); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1326 | llvm::Value *Callee = CGM.GetAddrOfCXXDestructor(DD, Type); | 
|  | 1327 |  | 
| Anders Carlsson | c997d42 | 2010-01-02 01:01:18 +0000 | [diff] [blame] | 1328 | EmitCXXMemberCall(DD, Callee, ReturnValueSlot(), This, VTT, 0, 0); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1329 | } | 
|  | 1330 |  | 
|  | 1331 | llvm::Value * | 
| Anders Carlsson | bb7e17b | 2010-01-31 01:36:53 +0000 | [diff] [blame] | 1332 | CodeGenFunction::GetVirtualBaseClassOffset(llvm::Value *This, | 
|  | 1333 | const CXXRecordDecl *ClassDecl, | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1334 | const CXXRecordDecl *BaseClassDecl) { | 
|  | 1335 | const llvm::Type *Int8PtrTy = | 
|  | 1336 | llvm::Type::getInt8Ty(VMContext)->getPointerTo(); | 
|  | 1337 |  | 
|  | 1338 | llvm::Value *VTablePtr = Builder.CreateBitCast(This, | 
|  | 1339 | Int8PtrTy->getPointerTo()); | 
|  | 1340 | VTablePtr = Builder.CreateLoad(VTablePtr, "vtable"); | 
|  | 1341 |  | 
| Anders Carlsson | bba1607 | 2010-03-11 07:15:17 +0000 | [diff] [blame] | 1342 | int64_t VBaseOffsetOffset = | 
| Anders Carlsson | af44035 | 2010-03-23 04:11:45 +0000 | [diff] [blame] | 1343 | CGM.getVTables().getVirtualBaseOffsetOffset(ClassDecl, BaseClassDecl); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1344 |  | 
|  | 1345 | llvm::Value *VBaseOffsetPtr = | 
| Anders Carlsson | bba1607 | 2010-03-11 07:15:17 +0000 | [diff] [blame] | 1346 | Builder.CreateConstGEP1_64(VTablePtr, VBaseOffsetOffset, "vbase.offset.ptr"); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1347 | const llvm::Type *PtrDiffTy = | 
|  | 1348 | ConvertType(getContext().getPointerDiffType()); | 
|  | 1349 |  | 
|  | 1350 | VBaseOffsetPtr = Builder.CreateBitCast(VBaseOffsetPtr, | 
|  | 1351 | PtrDiffTy->getPointerTo()); | 
|  | 1352 |  | 
|  | 1353 | llvm::Value *VBaseOffset = Builder.CreateLoad(VBaseOffsetPtr, "vbase.offset"); | 
|  | 1354 |  | 
|  | 1355 | return VBaseOffset; | 
|  | 1356 | } | 
|  | 1357 |  | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1358 | void | 
|  | 1359 | CodeGenFunction::InitializeVTablePointer(BaseSubobject Base, | 
| Anders Carlsson | b3b772e | 2010-04-20 05:22:15 +0000 | [diff] [blame] | 1360 | const CXXRecordDecl *NearestVBase, | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1361 | llvm::Constant *VTable, | 
|  | 1362 | const CXXRecordDecl *VTableClass) { | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1363 | const CXXRecordDecl *RD = Base.getBase(); | 
|  | 1364 |  | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1365 | // Compute the address point. | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1366 | llvm::Value *VTableAddressPoint; | 
| Anders Carlsson | 851853d | 2010-03-29 02:38:51 +0000 | [diff] [blame] | 1367 |  | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1368 | // Check if we need to use a vtable from the VTT. | 
| Anders Carlsson | 851853d | 2010-03-29 02:38:51 +0000 | [diff] [blame] | 1369 | if (CodeGenVTables::needsVTTParameter(CurGD) && | 
| Anders Carlsson | b3b772e | 2010-04-20 05:22:15 +0000 | [diff] [blame] | 1370 | (RD->getNumVBases() || NearestVBase)) { | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1371 | // Get the secondary vpointer index. | 
|  | 1372 | uint64_t VirtualPointerIndex = | 
|  | 1373 | CGM.getVTables().getSecondaryVirtualPointerIndex(VTableClass, Base); | 
|  | 1374 |  | 
|  | 1375 | /// Load the VTT. | 
|  | 1376 | llvm::Value *VTT = LoadCXXVTT(); | 
|  | 1377 | if (VirtualPointerIndex) | 
|  | 1378 | VTT = Builder.CreateConstInBoundsGEP1_64(VTT, VirtualPointerIndex); | 
|  | 1379 |  | 
|  | 1380 | // And load the address point from the VTT. | 
|  | 1381 | VTableAddressPoint = Builder.CreateLoad(VTT); | 
|  | 1382 | } else { | 
| Anders Carlsson | 64c9eca | 2010-03-29 02:08:26 +0000 | [diff] [blame] | 1383 | uint64_t AddressPoint = CGM.getVTables().getAddressPoint(Base, VTableClass); | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1384 | VTableAddressPoint = | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1385 | Builder.CreateConstInBoundsGEP2_64(VTable, 0, AddressPoint); | 
| Anders Carlsson | c83f106 | 2010-03-29 01:08:49 +0000 | [diff] [blame] | 1386 | } | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1387 |  | 
| Anders Carlsson | 36fd6be | 2010-04-20 16:22:16 +0000 | [diff] [blame] | 1388 | // Compute where to store the address point. | 
| Anders Carlsson | 3e79c30 | 2010-04-20 18:05:10 +0000 | [diff] [blame] | 1389 | llvm::Value *VTableField; | 
|  | 1390 |  | 
|  | 1391 | if (CodeGenVTables::needsVTTParameter(CurGD) && NearestVBase) { | 
|  | 1392 | // We need to use the virtual base offset offset because the virtual base | 
|  | 1393 | // might have a different offset in the most derived class. | 
| Anders Carlsson | a88ad56 | 2010-04-24 21:51:08 +0000 | [diff] [blame] | 1394 | VTableField = OldGetAddressOfBaseClass(LoadCXXThis(), VTableClass, RD); | 
| Anders Carlsson | 3e79c30 | 2010-04-20 18:05:10 +0000 | [diff] [blame] | 1395 | } else { | 
|  | 1396 | const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGM.getLLVMContext()); | 
| Anders Carlsson | 36fd6be | 2010-04-20 16:22:16 +0000 | [diff] [blame] | 1397 |  | 
| Anders Carlsson | 3e79c30 | 2010-04-20 18:05:10 +0000 | [diff] [blame] | 1398 | VTableField = Builder.CreateBitCast(LoadCXXThis(), Int8PtrTy); | 
| Anders Carlsson | 36fd6be | 2010-04-20 16:22:16 +0000 | [diff] [blame] | 1399 | VTableField = | 
| Anders Carlsson | 3e79c30 | 2010-04-20 18:05:10 +0000 | [diff] [blame] | 1400 | Builder.CreateConstInBoundsGEP1_64(VTableField, Base.getBaseOffset() / 8); | 
|  | 1401 | } | 
| Anders Carlsson | 36fd6be | 2010-04-20 16:22:16 +0000 | [diff] [blame] | 1402 |  | 
| Anders Carlsson | d103f9f | 2010-03-28 19:40:00 +0000 | [diff] [blame] | 1403 | // Finally, store the address point. | 
|  | 1404 | const llvm::Type *AddressPointPtrTy = | 
|  | 1405 | VTableAddressPoint->getType()->getPointerTo(); | 
|  | 1406 | VTableField = Builder.CreateBitCast(VTableField, AddressPointPtrTy); | 
|  | 1407 | Builder.CreateStore(VTableAddressPoint, VTableField); | 
|  | 1408 | } | 
|  | 1409 |  | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1410 | void | 
|  | 1411 | CodeGenFunction::InitializeVTablePointers(BaseSubobject Base, | 
| Anders Carlsson | b3b772e | 2010-04-20 05:22:15 +0000 | [diff] [blame] | 1412 | const CXXRecordDecl *NearestVBase, | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1413 | bool BaseIsNonVirtualPrimaryBase, | 
|  | 1414 | llvm::Constant *VTable, | 
|  | 1415 | const CXXRecordDecl *VTableClass, | 
|  | 1416 | VisitedVirtualBasesSetTy& VBases) { | 
|  | 1417 | // If this base is a non-virtual primary base the address point has already | 
|  | 1418 | // been set. | 
|  | 1419 | if (!BaseIsNonVirtualPrimaryBase) { | 
|  | 1420 | // Initialize the vtable pointer for this base. | 
| Anders Carlsson | 3e79c30 | 2010-04-20 18:05:10 +0000 | [diff] [blame] | 1421 | InitializeVTablePointer(Base, NearestVBase, VTable, VTableClass); | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1422 | } | 
|  | 1423 |  | 
|  | 1424 | const CXXRecordDecl *RD = Base.getBase(); | 
|  | 1425 |  | 
|  | 1426 | // Traverse bases. | 
|  | 1427 | for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(), | 
|  | 1428 | E = RD->bases_end(); I != E; ++I) { | 
|  | 1429 | CXXRecordDecl *BaseDecl | 
|  | 1430 | = cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl()); | 
|  | 1431 |  | 
|  | 1432 | // Ignore classes without a vtable. | 
|  | 1433 | if (!BaseDecl->isDynamicClass()) | 
|  | 1434 | continue; | 
|  | 1435 |  | 
|  | 1436 | uint64_t BaseOffset; | 
| Anders Carlsson | 14da9de | 2010-03-29 01:16:41 +0000 | [diff] [blame] | 1437 | bool BaseDeclIsNonVirtualPrimaryBase; | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1438 |  | 
|  | 1439 | if (I->isVirtual()) { | 
|  | 1440 | // Check if we've visited this virtual base before. | 
|  | 1441 | if (!VBases.insert(BaseDecl)) | 
|  | 1442 | continue; | 
|  | 1443 |  | 
|  | 1444 | const ASTRecordLayout &Layout = | 
|  | 1445 | getContext().getASTRecordLayout(VTableClass); | 
|  | 1446 |  | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1447 | BaseOffset = Layout.getVBaseClassOffset(BaseDecl); | 
| Anders Carlsson | 14da9de | 2010-03-29 01:16:41 +0000 | [diff] [blame] | 1448 | BaseDeclIsNonVirtualPrimaryBase = false; | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1449 | } else { | 
|  | 1450 | const ASTRecordLayout &Layout = getContext().getASTRecordLayout(RD); | 
|  | 1451 |  | 
|  | 1452 | BaseOffset = Base.getBaseOffset() + Layout.getBaseClassOffset(BaseDecl); | 
| Anders Carlsson | 14da9de | 2010-03-29 01:16:41 +0000 | [diff] [blame] | 1453 | BaseDeclIsNonVirtualPrimaryBase = Layout.getPrimaryBase() == BaseDecl; | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1454 | } | 
|  | 1455 |  | 
|  | 1456 | InitializeVTablePointers(BaseSubobject(BaseDecl, BaseOffset), | 
| Anders Carlsson | b3b772e | 2010-04-20 05:22:15 +0000 | [diff] [blame] | 1457 | I->isVirtual() ? BaseDecl : NearestVBase, | 
| Anders Carlsson | 14da9de | 2010-03-29 01:16:41 +0000 | [diff] [blame] | 1458 | BaseDeclIsNonVirtualPrimaryBase, | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1459 | VTable, VTableClass, VBases); | 
|  | 1460 | } | 
|  | 1461 | } | 
|  | 1462 |  | 
|  | 1463 | void CodeGenFunction::InitializeVTablePointers(const CXXRecordDecl *RD) { | 
|  | 1464 | // Ignore classes without a vtable. | 
| Anders Carlsson | 0703690 | 2010-03-26 04:39:42 +0000 | [diff] [blame] | 1465 | if (!RD->isDynamicClass()) | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1466 | return; | 
|  | 1467 |  | 
| Anders Carlsson | 0703690 | 2010-03-26 04:39:42 +0000 | [diff] [blame] | 1468 | // Get the VTable. | 
|  | 1469 | llvm::Constant *VTable = CGM.getVTables().GetAddrOfVTable(RD); | 
| Anders Carlsson | 5c6c1d9 | 2010-03-24 03:57:14 +0000 | [diff] [blame] | 1470 |  | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1471 | // Initialize the vtable pointers for this class and all of its bases. | 
|  | 1472 | VisitedVirtualBasesSetTy VBases; | 
| Anders Carlsson | b3b772e | 2010-04-20 05:22:15 +0000 | [diff] [blame] | 1473 | InitializeVTablePointers(BaseSubobject(RD, 0), /*NearestVBase=*/0, | 
| Anders Carlsson | 603d6d1 | 2010-03-28 21:07:49 +0000 | [diff] [blame] | 1474 | /*BaseIsNonVirtualPrimaryBase=*/false, | 
|  | 1475 | VTable, RD, VBases); | 
| Anders Carlsson | 3b5ad22 | 2010-01-01 20:29:01 +0000 | [diff] [blame] | 1476 | } |