2009-11-24 13:51:11 +08:00
|
|
|
//===--- CGClass.cpp - Emit LLVM Code for C++ classes ---------------------===//
|
2009-09-12 12:27:24 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This contains code dealing with C++ code generation of classes
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-12 05:04:37 +08:00
|
|
|
#include "CGDebugInfo.h"
|
2009-09-12 12:27:24 +08:00
|
|
|
#include "CodeGenFunction.h"
|
2009-10-07 06:43:30 +08:00
|
|
|
#include "clang/AST/CXXInheritance.h"
|
2010-09-17 10:31:44 +08:00
|
|
|
#include "clang/AST/EvaluatedExprVisitor.h"
|
2009-09-12 12:27:24 +08:00
|
|
|
#include "clang/AST/RecordLayout.h"
|
2010-02-19 17:25:03 +08:00
|
|
|
#include "clang/AST/StmtCXX.h"
|
2011-02-23 04:55:26 +08:00
|
|
|
#include "clang/Frontend/CodeGenOptions.h"
|
2009-10-07 06:43:30 +08:00
|
|
|
|
2009-09-12 12:27:24 +08:00
|
|
|
using namespace clang;
|
|
|
|
using namespace CodeGen;
|
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
static CharUnits
|
2010-04-25 05:06:20 +08:00
|
|
|
ComputeNonVirtualBaseClassOffset(ASTContext &Context,
|
|
|
|
const CXXRecordDecl *DerivedClass,
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_const_iterator Start,
|
|
|
|
CastExpr::path_const_iterator End) {
|
2011-03-22 08:53:26 +08:00
|
|
|
CharUnits Offset = CharUnits::Zero();
|
2010-04-25 05:06:20 +08:00
|
|
|
|
|
|
|
const CXXRecordDecl *RD = DerivedClass;
|
|
|
|
|
2010-08-07 14:22:56 +08:00
|
|
|
for (CastExpr::path_const_iterator I = Start; I != End; ++I) {
|
2010-04-25 05:06:20 +08:00
|
|
|
const CXXBaseSpecifier *Base = *I;
|
|
|
|
assert(!Base->isVirtual() && "Should not see virtual bases here!");
|
|
|
|
|
|
|
|
// Get the layout.
|
|
|
|
const ASTRecordLayout &Layout = Context.getASTRecordLayout(RD);
|
|
|
|
|
|
|
|
const CXXRecordDecl *BaseDecl =
|
|
|
|
cast<CXXRecordDecl>(Base->getType()->getAs<RecordType>()->getDecl());
|
|
|
|
|
|
|
|
// Add the offset.
|
2011-03-22 08:53:26 +08:00
|
|
|
Offset += Layout.getBaseClassOffset(BaseDecl);
|
2010-04-25 05:06:20 +08:00
|
|
|
|
|
|
|
RD = BaseDecl;
|
|
|
|
}
|
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
return Offset;
|
2010-04-25 05:06:20 +08:00
|
|
|
}
|
2009-09-12 12:27:24 +08:00
|
|
|
|
2010-04-25 05:23:59 +08:00
|
|
|
llvm::Constant *
|
|
|
|
CodeGenModule::GetNonVirtualBaseClassOffset(const CXXRecordDecl *ClassDecl,
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_const_iterator PathBegin,
|
|
|
|
CastExpr::path_const_iterator PathEnd) {
|
|
|
|
assert(PathBegin != PathEnd && "Base path should not be empty!");
|
2010-04-25 05:23:59 +08:00
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
CharUnits Offset =
|
2010-08-07 14:22:56 +08:00
|
|
|
ComputeNonVirtualBaseClassOffset(getContext(), ClassDecl,
|
|
|
|
PathBegin, PathEnd);
|
2011-03-22 08:53:26 +08:00
|
|
|
if (Offset.isZero())
|
2010-04-25 05:23:59 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
const llvm::Type *PtrDiffTy =
|
|
|
|
Types.ConvertType(getContext().getPointerDiffType());
|
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
return llvm::ConstantInt::get(PtrDiffTy, Offset.getQuantity());
|
2009-09-29 11:13:20 +08:00
|
|
|
}
|
|
|
|
|
2010-04-25 07:01:49 +08:00
|
|
|
/// Gets the address of a direct base class within a complete object.
|
2010-02-16 12:15:37 +08:00
|
|
|
/// This should only be used for (1) non-virtual bases or (2) virtual bases
|
|
|
|
/// when the type is known to be complete (e.g. in complete destructors).
|
|
|
|
///
|
|
|
|
/// The object pointed to by 'This' is assumed to be non-null.
|
|
|
|
llvm::Value *
|
2010-04-25 07:01:49 +08:00
|
|
|
CodeGenFunction::GetAddressOfDirectBaseInCompleteClass(llvm::Value *This,
|
|
|
|
const CXXRecordDecl *Derived,
|
|
|
|
const CXXRecordDecl *Base,
|
|
|
|
bool BaseIsVirtual) {
|
2010-02-16 12:15:37 +08:00
|
|
|
// 'this' must be a pointer (in some address space) to Derived.
|
|
|
|
assert(This->getType()->isPointerTy() &&
|
|
|
|
cast<llvm::PointerType>(This->getType())->getElementType()
|
|
|
|
== ConvertType(Derived));
|
|
|
|
|
|
|
|
// Compute the offset of the virtual base.
|
2011-03-22 09:21:15 +08:00
|
|
|
CharUnits Offset;
|
2010-02-16 12:15:37 +08:00
|
|
|
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(Derived);
|
2010-04-25 07:01:49 +08:00
|
|
|
if (BaseIsVirtual)
|
2011-03-22 09:21:15 +08:00
|
|
|
Offset = Layout.getVBaseClassOffset(Base);
|
2010-02-16 12:15:37 +08:00
|
|
|
else
|
2011-03-22 09:21:15 +08:00
|
|
|
Offset = Layout.getBaseClassOffset(Base);
|
2010-02-16 12:15:37 +08:00
|
|
|
|
|
|
|
// Shift and cast down to the base type.
|
|
|
|
// TODO: for complete types, this should be possible with a GEP.
|
|
|
|
llvm::Value *V = This;
|
2011-03-22 09:21:15 +08:00
|
|
|
if (Offset.isPositive()) {
|
2010-02-16 12:15:37 +08:00
|
|
|
const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(getLLVMContext());
|
|
|
|
V = Builder.CreateBitCast(V, Int8PtrTy);
|
2011-03-22 09:21:15 +08:00
|
|
|
V = Builder.CreateConstInBoundsGEP1_64(V, Offset.getQuantity());
|
2010-02-16 12:15:37 +08:00
|
|
|
}
|
|
|
|
V = Builder.CreateBitCast(V, ConvertType(Base)->getPointerTo());
|
|
|
|
|
|
|
|
return V;
|
2010-03-29 03:40:00 +08:00
|
|
|
}
|
2010-02-16 12:15:37 +08:00
|
|
|
|
2010-04-21 00:03:35 +08:00
|
|
|
static llvm::Value *
|
|
|
|
ApplyNonVirtualAndVirtualOffset(CodeGenFunction &CGF, llvm::Value *ThisPtr,
|
2011-03-23 08:45:26 +08:00
|
|
|
CharUnits NonVirtual, llvm::Value *Virtual) {
|
2010-04-21 00:03:35 +08:00
|
|
|
const llvm::Type *PtrDiffTy =
|
|
|
|
CGF.ConvertType(CGF.getContext().getPointerDiffType());
|
|
|
|
|
|
|
|
llvm::Value *NonVirtualOffset = 0;
|
2011-03-23 08:45:26 +08:00
|
|
|
if (!NonVirtual.isZero())
|
|
|
|
NonVirtualOffset = llvm::ConstantInt::get(PtrDiffTy,
|
|
|
|
NonVirtual.getQuantity());
|
2010-04-21 00:03:35 +08:00
|
|
|
|
|
|
|
llvm::Value *BaseOffset;
|
|
|
|
if (Virtual) {
|
|
|
|
if (NonVirtualOffset)
|
|
|
|
BaseOffset = CGF.Builder.CreateAdd(Virtual, NonVirtualOffset);
|
|
|
|
else
|
|
|
|
BaseOffset = Virtual;
|
|
|
|
} else
|
|
|
|
BaseOffset = NonVirtualOffset;
|
|
|
|
|
|
|
|
// Apply the base offset.
|
|
|
|
const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGF.getLLVMContext());
|
|
|
|
ThisPtr = CGF.Builder.CreateBitCast(ThisPtr, Int8PtrTy);
|
|
|
|
ThisPtr = CGF.Builder.CreateGEP(ThisPtr, BaseOffset, "add.ptr");
|
|
|
|
|
|
|
|
return ThisPtr;
|
|
|
|
}
|
|
|
|
|
2010-04-25 05:06:20 +08:00
|
|
|
llvm::Value *
|
|
|
|
CodeGenFunction::GetAddressOfBaseClass(llvm::Value *Value,
|
2010-04-25 07:01:49 +08:00
|
|
|
const CXXRecordDecl *Derived,
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_const_iterator PathBegin,
|
|
|
|
CastExpr::path_const_iterator PathEnd,
|
2010-04-25 05:06:20 +08:00
|
|
|
bool NullCheckValue) {
|
2010-08-07 14:22:56 +08:00
|
|
|
assert(PathBegin != PathEnd && "Base path should not be empty!");
|
2010-04-25 05:06:20 +08:00
|
|
|
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_const_iterator Start = PathBegin;
|
2010-04-25 05:06:20 +08:00
|
|
|
const CXXRecordDecl *VBase = 0;
|
|
|
|
|
|
|
|
// Get the virtual base.
|
|
|
|
if ((*Start)->isVirtual()) {
|
|
|
|
VBase =
|
|
|
|
cast<CXXRecordDecl>((*Start)->getType()->getAs<RecordType>()->getDecl());
|
|
|
|
++Start;
|
|
|
|
}
|
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
CharUnits NonVirtualOffset =
|
2010-04-25 07:01:49 +08:00
|
|
|
ComputeNonVirtualBaseClassOffset(getContext(), VBase ? VBase : Derived,
|
2010-08-07 14:22:56 +08:00
|
|
|
Start, PathEnd);
|
2010-04-25 05:06:20 +08:00
|
|
|
|
|
|
|
// Get the base pointer type.
|
|
|
|
const llvm::Type *BasePtrTy =
|
2010-08-07 14:22:56 +08:00
|
|
|
ConvertType((PathEnd[-1])->getType())->getPointerTo();
|
2010-04-25 05:06:20 +08:00
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
if (NonVirtualOffset.isZero() && !VBase) {
|
2010-04-25 05:06:20 +08:00
|
|
|
// Just cast back.
|
|
|
|
return Builder.CreateBitCast(Value, BasePtrTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::BasicBlock *CastNull = 0;
|
|
|
|
llvm::BasicBlock *CastNotNull = 0;
|
|
|
|
llvm::BasicBlock *CastEnd = 0;
|
|
|
|
|
|
|
|
if (NullCheckValue) {
|
|
|
|
CastNull = createBasicBlock("cast.null");
|
|
|
|
CastNotNull = createBasicBlock("cast.notnull");
|
|
|
|
CastEnd = createBasicBlock("cast.end");
|
|
|
|
|
2011-04-11 08:30:07 +08:00
|
|
|
llvm::Value *IsNull = Builder.CreateIsNull(Value);
|
2010-04-25 05:06:20 +08:00
|
|
|
Builder.CreateCondBr(IsNull, CastNull, CastNotNull);
|
|
|
|
EmitBlock(CastNotNull);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Value *VirtualOffset = 0;
|
|
|
|
|
2011-01-29 11:18:56 +08:00
|
|
|
if (VBase) {
|
|
|
|
if (Derived->hasAttr<FinalAttr>()) {
|
|
|
|
VirtualOffset = 0;
|
|
|
|
|
|
|
|
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(Derived);
|
|
|
|
|
2011-03-22 08:53:26 +08:00
|
|
|
CharUnits VBaseOffset = Layout.getVBaseClassOffset(VBase);
|
|
|
|
NonVirtualOffset += VBaseOffset;
|
2011-01-29 11:18:56 +08:00
|
|
|
} else
|
|
|
|
VirtualOffset = GetVirtualBaseClassOffset(Value, Derived, VBase);
|
|
|
|
}
|
2010-04-25 05:06:20 +08:00
|
|
|
|
|
|
|
// Apply the offsets.
|
2011-03-22 08:53:26 +08:00
|
|
|
Value = ApplyNonVirtualAndVirtualOffset(*this, Value,
|
2011-03-23 08:45:26 +08:00
|
|
|
NonVirtualOffset,
|
2010-04-25 05:06:20 +08:00
|
|
|
VirtualOffset);
|
|
|
|
|
|
|
|
// Cast back.
|
|
|
|
Value = Builder.CreateBitCast(Value, BasePtrTy);
|
|
|
|
|
|
|
|
if (NullCheckValue) {
|
|
|
|
Builder.CreateBr(CastEnd);
|
|
|
|
EmitBlock(CastNull);
|
|
|
|
Builder.CreateBr(CastEnd);
|
|
|
|
EmitBlock(CastEnd);
|
|
|
|
|
2011-03-30 19:28:58 +08:00
|
|
|
llvm::PHINode *PHI = Builder.CreatePHI(Value->getType(), 2);
|
2010-04-25 05:06:20 +08:00
|
|
|
PHI->addIncoming(Value, CastNotNull);
|
|
|
|
PHI->addIncoming(llvm::Constant::getNullValue(Value->getType()),
|
|
|
|
CastNull);
|
|
|
|
Value = PHI;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
2009-11-24 01:57:54 +08:00
|
|
|
llvm::Value *
|
|
|
|
CodeGenFunction::GetAddressOfDerivedClass(llvm::Value *Value,
|
2010-04-25 07:01:49 +08:00
|
|
|
const CXXRecordDecl *Derived,
|
2010-08-07 14:22:56 +08:00
|
|
|
CastExpr::path_const_iterator PathBegin,
|
|
|
|
CastExpr::path_const_iterator PathEnd,
|
2009-11-24 01:57:54 +08:00
|
|
|
bool NullCheckValue) {
|
2010-08-07 14:22:56 +08:00
|
|
|
assert(PathBegin != PathEnd && "Base path should not be empty!");
|
2010-04-25 05:23:59 +08:00
|
|
|
|
2009-11-24 01:57:54 +08:00
|
|
|
QualType DerivedTy =
|
2010-04-25 07:01:49 +08:00
|
|
|
getContext().getCanonicalType(getContext().getTagDeclType(Derived));
|
2009-11-24 01:57:54 +08:00
|
|
|
const llvm::Type *DerivedPtrTy = ConvertType(DerivedTy)->getPointerTo();
|
|
|
|
|
2010-01-31 09:43:37 +08:00
|
|
|
llvm::Value *NonVirtualOffset =
|
2010-08-07 14:22:56 +08:00
|
|
|
CGM.GetNonVirtualBaseClassOffset(Derived, PathBegin, PathEnd);
|
2010-01-31 09:43:37 +08:00
|
|
|
|
|
|
|
if (!NonVirtualOffset) {
|
|
|
|
// No offset, we can just cast back.
|
|
|
|
return Builder.CreateBitCast(Value, DerivedPtrTy);
|
|
|
|
}
|
|
|
|
|
2009-11-24 01:57:54 +08:00
|
|
|
llvm::BasicBlock *CastNull = 0;
|
|
|
|
llvm::BasicBlock *CastNotNull = 0;
|
|
|
|
llvm::BasicBlock *CastEnd = 0;
|
|
|
|
|
|
|
|
if (NullCheckValue) {
|
|
|
|
CastNull = createBasicBlock("cast.null");
|
|
|
|
CastNotNull = createBasicBlock("cast.notnull");
|
|
|
|
CastEnd = createBasicBlock("cast.end");
|
|
|
|
|
2011-04-11 08:30:07 +08:00
|
|
|
llvm::Value *IsNull = Builder.CreateIsNull(Value);
|
2009-11-24 01:57:54 +08:00
|
|
|
Builder.CreateCondBr(IsNull, CastNull, CastNotNull);
|
|
|
|
EmitBlock(CastNotNull);
|
|
|
|
}
|
|
|
|
|
2010-01-31 09:43:37 +08:00
|
|
|
// Apply the offset.
|
|
|
|
Value = Builder.CreatePtrToInt(Value, NonVirtualOffset->getType());
|
|
|
|
Value = Builder.CreateSub(Value, NonVirtualOffset);
|
|
|
|
Value = Builder.CreateIntToPtr(Value, DerivedPtrTy);
|
|
|
|
|
|
|
|
// Just cast.
|
|
|
|
Value = Builder.CreateBitCast(Value, DerivedPtrTy);
|
2009-11-24 01:57:54 +08:00
|
|
|
|
|
|
|
if (NullCheckValue) {
|
|
|
|
Builder.CreateBr(CastEnd);
|
|
|
|
EmitBlock(CastNull);
|
|
|
|
Builder.CreateBr(CastEnd);
|
|
|
|
EmitBlock(CastEnd);
|
|
|
|
|
2011-03-30 19:28:58 +08:00
|
|
|
llvm::PHINode *PHI = Builder.CreatePHI(Value->getType(), 2);
|
2009-11-24 01:57:54 +08:00
|
|
|
PHI->addIncoming(Value, CastNotNull);
|
|
|
|
PHI->addIncoming(llvm::Constant::getNullValue(Value->getType()),
|
2009-09-12 14:04:24 +08:00
|
|
|
CastNull);
|
2009-11-24 01:57:54 +08:00
|
|
|
Value = PHI;
|
2009-09-12 14:04:24 +08:00
|
|
|
}
|
2009-09-12 12:27:24 +08:00
|
|
|
|
2009-11-24 01:57:54 +08:00
|
|
|
return Value;
|
2009-09-12 12:27:24 +08:00
|
|
|
}
|
2010-03-30 11:27:09 +08:00
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
/// GetVTTParameter - Return the VTT parameter that should be passed to a
|
|
|
|
/// base constructor/destructor with virtual bases.
|
2010-05-03 07:33:10 +08:00
|
|
|
static llvm::Value *GetVTTParameter(CodeGenFunction &CGF, GlobalDecl GD,
|
|
|
|
bool ForVirtualBase) {
|
2010-03-23 12:11:45 +08:00
|
|
|
if (!CodeGenVTables::needsVTTParameter(GD)) {
|
2010-01-02 09:01:18 +08:00
|
|
|
// This constructor/destructor does not need a VTT parameter.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const CXXRecordDecl *RD = cast<CXXMethodDecl>(CGF.CurFuncDecl)->getParent();
|
|
|
|
const CXXRecordDecl *Base = cast<CXXMethodDecl>(GD.getDecl())->getParent();
|
2010-02-19 03:59:28 +08:00
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
llvm::Value *VTT;
|
|
|
|
|
2010-02-19 03:59:28 +08:00
|
|
|
uint64_t SubVTTIndex;
|
|
|
|
|
|
|
|
// If the record matches the base, this is the complete ctor/dtor
|
|
|
|
// variant calling the base variant in a class with virtual bases.
|
|
|
|
if (RD == Base) {
|
2010-03-23 12:11:45 +08:00
|
|
|
assert(!CodeGenVTables::needsVTTParameter(CGF.CurGD) &&
|
2010-02-19 03:59:28 +08:00
|
|
|
"doing no-op VTT offset in base dtor/ctor?");
|
2010-05-03 07:33:10 +08:00
|
|
|
assert(!ForVirtualBase && "Can't have same class as virtual base!");
|
2010-02-19 03:59:28 +08:00
|
|
|
SubVTTIndex = 0;
|
|
|
|
} else {
|
2010-05-03 07:53:25 +08:00
|
|
|
const ASTRecordLayout &Layout =
|
|
|
|
CGF.getContext().getASTRecordLayout(RD);
|
2011-03-24 09:21:01 +08:00
|
|
|
CharUnits BaseOffset = ForVirtualBase ?
|
|
|
|
Layout.getVBaseClassOffset(Base) :
|
|
|
|
Layout.getBaseClassOffset(Base);
|
2010-05-03 07:53:25 +08:00
|
|
|
|
|
|
|
SubVTTIndex =
|
|
|
|
CGF.CGM.getVTables().getSubVTTIndex(RD, BaseSubobject(Base, BaseOffset));
|
2010-02-19 03:59:28 +08:00
|
|
|
assert(SubVTTIndex != 0 && "Sub-VTT index must be greater than zero!");
|
|
|
|
}
|
2010-01-02 09:01:18 +08:00
|
|
|
|
2010-03-23 12:11:45 +08:00
|
|
|
if (CodeGenVTables::needsVTTParameter(CGF.CurGD)) {
|
2010-01-02 09:01:18 +08:00
|
|
|
// A VTT parameter was passed to the constructor, use it.
|
|
|
|
VTT = CGF.LoadCXXVTT();
|
|
|
|
VTT = CGF.Builder.CreateConstInBoundsGEP1_64(VTT, SubVTTIndex);
|
|
|
|
} else {
|
|
|
|
// We're the complete constructor, so get the VTT by name.
|
2011-01-30 03:16:51 +08:00
|
|
|
VTT = CGF.CGM.getVTables().GetAddrOfVTT(RD);
|
2010-01-02 09:01:18 +08:00
|
|
|
VTT = CGF.Builder.CreateConstInBoundsGEP2_64(VTT, 0, SubVTTIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return VTT;
|
|
|
|
}
|
|
|
|
|
2010-07-21 09:23:41 +08:00
|
|
|
namespace {
|
2010-07-21 13:30:47 +08:00
|
|
|
/// Call the destructor for a direct base class.
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallBaseDtor : EHScopeStack::Cleanup {
|
2010-07-21 13:30:47 +08:00
|
|
|
const CXXRecordDecl *BaseClass;
|
|
|
|
bool BaseIsVirtual;
|
|
|
|
CallBaseDtor(const CXXRecordDecl *Base, bool BaseIsVirtual)
|
|
|
|
: BaseClass(Base), BaseIsVirtual(BaseIsVirtual) {}
|
2010-07-21 09:23:41 +08:00
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
2010-07-21 13:30:47 +08:00
|
|
|
const CXXRecordDecl *DerivedClass =
|
|
|
|
cast<CXXMethodDecl>(CGF.CurCodeDecl)->getParent();
|
|
|
|
|
|
|
|
const CXXDestructorDecl *D = BaseClass->getDestructor();
|
|
|
|
llvm::Value *Addr =
|
|
|
|
CGF.GetAddressOfDirectBaseInCompleteClass(CGF.LoadCXXThis(),
|
|
|
|
DerivedClass, BaseClass,
|
|
|
|
BaseIsVirtual);
|
|
|
|
CGF.EmitCXXDestructorCall(D, Dtor_Base, BaseIsVirtual, Addr);
|
2010-07-21 09:23:41 +08:00
|
|
|
}
|
|
|
|
};
|
2010-09-17 10:31:44 +08:00
|
|
|
|
|
|
|
/// A visitor which checks whether an initializer uses 'this' in a
|
|
|
|
/// way which requires the vtable to be properly set.
|
|
|
|
struct DynamicThisUseChecker : EvaluatedExprVisitor<DynamicThisUseChecker> {
|
|
|
|
typedef EvaluatedExprVisitor<DynamicThisUseChecker> super;
|
|
|
|
|
|
|
|
bool UsesThis;
|
|
|
|
|
|
|
|
DynamicThisUseChecker(ASTContext &C) : super(C), UsesThis(false) {}
|
|
|
|
|
|
|
|
// Black-list all explicit and implicit references to 'this'.
|
|
|
|
//
|
|
|
|
// Do we need to worry about external references to 'this' derived
|
|
|
|
// from arbitrary code? If so, then anything which runs arbitrary
|
|
|
|
// external code might potentially access the vtable.
|
|
|
|
void VisitCXXThisExpr(CXXThisExpr *E) { UsesThis = true; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool BaseInitializerUsesThis(ASTContext &C, const Expr *Init) {
|
|
|
|
DynamicThisUseChecker Checker(C);
|
|
|
|
Checker.Visit(const_cast<Expr*>(Init));
|
|
|
|
return Checker.UsesThis;
|
2010-07-21 09:23:41 +08:00
|
|
|
}
|
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
static void EmitBaseInitializer(CodeGenFunction &CGF,
|
|
|
|
const CXXRecordDecl *ClassDecl,
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *BaseInit,
|
2009-12-25 06:46:43 +08:00
|
|
|
CXXCtorType CtorType) {
|
|
|
|
assert(BaseInit->isBaseInitializer() &&
|
|
|
|
"Must have base initializer!");
|
|
|
|
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
|
|
|
|
|
|
const Type *BaseType = BaseInit->getBaseClass();
|
|
|
|
CXXRecordDecl *BaseClassDecl =
|
|
|
|
cast<CXXRecordDecl>(BaseType->getAs<RecordType>()->getDecl());
|
|
|
|
|
2010-04-12 08:51:03 +08:00
|
|
|
bool isBaseVirtual = BaseInit->isBaseVirtual();
|
2009-12-25 06:46:43 +08:00
|
|
|
|
|
|
|
// The base constructor doesn't construct virtual bases.
|
|
|
|
if (CtorType == Ctor_Base && isBaseVirtual)
|
|
|
|
return;
|
|
|
|
|
2010-09-17 10:31:44 +08:00
|
|
|
// If the initializer for the base (other than the constructor
|
|
|
|
// itself) accesses 'this' in any way, we need to initialize the
|
|
|
|
// vtables.
|
|
|
|
if (BaseInitializerUsesThis(CGF.getContext(), BaseInit->getInit()))
|
|
|
|
CGF.InitializeVTablePointers(ClassDecl);
|
|
|
|
|
2010-02-16 12:15:37 +08:00
|
|
|
// We can pretend to be a complete class because it only matters for
|
|
|
|
// virtual bases, and we only do virtual bases for complete ctors.
|
2010-04-25 07:01:49 +08:00
|
|
|
llvm::Value *V =
|
|
|
|
CGF.GetAddressOfDirectBaseInCompleteClass(ThisPtr, ClassDecl,
|
2010-07-21 13:30:47 +08:00
|
|
|
BaseClassDecl,
|
|
|
|
isBaseVirtual);
|
2010-02-16 12:15:37 +08:00
|
|
|
|
2010-09-15 18:14:12 +08:00
|
|
|
AggValueSlot AggSlot = AggValueSlot::forAddr(V, false, /*Lifetime*/ true);
|
|
|
|
|
|
|
|
CGF.EmitAggExpr(BaseInit->getInit(), AggSlot);
|
2010-02-07 04:00:21 +08:00
|
|
|
|
2011-02-28 08:33:03 +08:00
|
|
|
if (CGF.CGM.getLangOptions().Exceptions &&
|
2011-02-20 08:20:27 +08:00
|
|
|
!BaseClassDecl->hasTrivialDestructor())
|
2010-07-21 15:22:38 +08:00
|
|
|
CGF.EHStack.pushCleanup<CallBaseDtor>(EHCleanup, BaseClassDecl,
|
|
|
|
isBaseVirtual);
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
static void EmitAggMemberInitializer(CodeGenFunction &CGF,
|
|
|
|
LValue LHS,
|
|
|
|
llvm::Value *ArrayIndexVar,
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *MemberInit,
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
QualType T,
|
|
|
|
unsigned Index) {
|
|
|
|
if (Index == MemberInit->getNumArrayIndices()) {
|
2010-07-06 09:34:17 +08:00
|
|
|
CodeGenFunction::RunCleanupsScope Cleanups(CGF);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
|
|
|
llvm::Value *Dest = LHS.getAddress();
|
|
|
|
if (ArrayIndexVar) {
|
|
|
|
// If we have an array index variable, load it and use it as an offset.
|
|
|
|
// Then, increment the value.
|
|
|
|
llvm::Value *ArrayIndex = CGF.Builder.CreateLoad(ArrayIndexVar);
|
|
|
|
Dest = CGF.Builder.CreateInBoundsGEP(Dest, ArrayIndex, "destaddress");
|
|
|
|
llvm::Value *Next = llvm::ConstantInt::get(ArrayIndex->getType(), 1);
|
|
|
|
Next = CGF.Builder.CreateAdd(ArrayIndex, Next, "inc");
|
|
|
|
CGF.Builder.CreateStore(Next, ArrayIndexVar);
|
|
|
|
}
|
2010-09-15 18:14:12 +08:00
|
|
|
|
|
|
|
AggValueSlot Slot = AggValueSlot::forAddr(Dest, LHS.isVolatileQualified(),
|
|
|
|
/*Lifetime*/ true);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
2010-09-15 18:14:12 +08:00
|
|
|
CGF.EmitAggExpr(MemberInit->getInit(), Slot);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ConstantArrayType *Array = CGF.getContext().getAsConstantArrayType(T);
|
|
|
|
assert(Array && "Array initialization without the array type?");
|
|
|
|
llvm::Value *IndexVar
|
|
|
|
= CGF.GetAddrOfLocalVar(MemberInit->getArrayIndex(Index));
|
|
|
|
assert(IndexVar && "Array index variable not loaded");
|
|
|
|
|
|
|
|
// Initialize this index variable to zero.
|
|
|
|
llvm::Value* Zero
|
|
|
|
= llvm::Constant::getNullValue(
|
|
|
|
CGF.ConvertType(CGF.getContext().getSizeType()));
|
|
|
|
CGF.Builder.CreateStore(Zero, IndexVar);
|
|
|
|
|
|
|
|
// Start the loop with a block that tests the condition.
|
|
|
|
llvm::BasicBlock *CondBlock = CGF.createBasicBlock("for.cond");
|
|
|
|
llvm::BasicBlock *AfterFor = CGF.createBasicBlock("for.end");
|
|
|
|
|
|
|
|
CGF.EmitBlock(CondBlock);
|
|
|
|
|
|
|
|
llvm::BasicBlock *ForBody = CGF.createBasicBlock("for.body");
|
|
|
|
// Generate: if (loop-index < number-of-elements) fall to the loop body,
|
|
|
|
// otherwise, go to the block after the for-loop.
|
|
|
|
uint64_t NumElements = Array->getSize().getZExtValue();
|
|
|
|
llvm::Value *Counter = CGF.Builder.CreateLoad(IndexVar);
|
2010-05-06 14:35:23 +08:00
|
|
|
llvm::Value *NumElementsPtr =
|
|
|
|
llvm::ConstantInt::get(Counter->getType(), NumElements);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
llvm::Value *IsLess = CGF.Builder.CreateICmpULT(Counter, NumElementsPtr,
|
|
|
|
"isless");
|
|
|
|
|
|
|
|
// If the condition is true, execute the body.
|
|
|
|
CGF.Builder.CreateCondBr(IsLess, ForBody, AfterFor);
|
|
|
|
|
|
|
|
CGF.EmitBlock(ForBody);
|
|
|
|
llvm::BasicBlock *ContinueBlock = CGF.createBasicBlock("for.inc");
|
|
|
|
|
|
|
|
{
|
2010-07-06 09:34:17 +08:00
|
|
|
CodeGenFunction::RunCleanupsScope Cleanups(CGF);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
|
|
|
// Inside the loop body recurse to emit the inner loop or, eventually, the
|
|
|
|
// constructor call.
|
|
|
|
EmitAggMemberInitializer(CGF, LHS, ArrayIndexVar, MemberInit,
|
|
|
|
Array->getElementType(), Index + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
CGF.EmitBlock(ContinueBlock);
|
|
|
|
|
|
|
|
// Emit the increment of the loop counter.
|
|
|
|
llvm::Value *NextVal = llvm::ConstantInt::get(Counter->getType(), 1);
|
|
|
|
Counter = CGF.Builder.CreateLoad(IndexVar);
|
|
|
|
NextVal = CGF.Builder.CreateAdd(Counter, NextVal, "inc");
|
|
|
|
CGF.Builder.CreateStore(NextVal, IndexVar);
|
|
|
|
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
|
|
CGF.EmitBranch(CondBlock);
|
|
|
|
|
|
|
|
// Emit the fall-through block.
|
|
|
|
CGF.EmitBlock(AfterFor, true);
|
|
|
|
}
|
2010-07-21 09:23:41 +08:00
|
|
|
|
|
|
|
namespace {
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallMemberDtor : EHScopeStack::Cleanup {
|
2010-07-21 09:23:41 +08:00
|
|
|
FieldDecl *Field;
|
|
|
|
CXXDestructorDecl *Dtor;
|
|
|
|
|
|
|
|
CallMemberDtor(FieldDecl *Field, CXXDestructorDecl *Dtor)
|
|
|
|
: Field(Field), Dtor(Dtor) {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
// FIXME: Is this OK for C++0x delegating constructors?
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
|
|
LValue LHS = CGF.EmitLValueForField(ThisPtr, Field, 0);
|
|
|
|
|
|
|
|
CGF.EmitCXXDestructorCall(Dtor, Dtor_Complete, /*ForVirtualBase=*/false,
|
|
|
|
LHS.getAddress());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
static void EmitMemberInitializer(CodeGenFunction &CGF,
|
|
|
|
const CXXRecordDecl *ClassDecl,
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *MemberInit,
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
const CXXConstructorDecl *Constructor,
|
|
|
|
FunctionArgList &Args) {
|
2010-12-04 17:14:42 +08:00
|
|
|
assert(MemberInit->isAnyMemberInitializer() &&
|
2009-12-25 06:46:43 +08:00
|
|
|
"Must have member initializer!");
|
|
|
|
|
|
|
|
// non-static data member initializers.
|
2010-12-04 17:14:42 +08:00
|
|
|
FieldDecl *Field = MemberInit->getAnyMember();
|
2009-12-25 06:46:43 +08:00
|
|
|
QualType FieldType = CGF.getContext().getCanonicalType(Field->getType());
|
|
|
|
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
2010-05-21 09:18:57 +08:00
|
|
|
LValue LHS;
|
2010-01-29 13:24:29 +08:00
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
// If we are initializing an anonymous union field, drill down to the field.
|
2010-12-04 17:14:42 +08:00
|
|
|
if (MemberInit->isIndirectMemberInitializer()) {
|
|
|
|
LHS = CGF.EmitLValueForAnonRecordField(ThisPtr,
|
|
|
|
MemberInit->getIndirectMember(), 0);
|
|
|
|
FieldType = MemberInit->getIndirectMember()->getAnonField()->getType();
|
2010-05-21 09:18:57 +08:00
|
|
|
} else {
|
|
|
|
LHS = CGF.EmitLValueForFieldInitialization(ThisPtr, Field, 0);
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
// FIXME: If there's no initializer and the CXXCtorInitializer
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
// was implicitly generated, we shouldn't be zeroing memory.
|
2009-12-25 06:46:43 +08:00
|
|
|
RValue RHS;
|
|
|
|
if (FieldType->isReferenceType()) {
|
2010-06-27 00:35:32 +08:00
|
|
|
RHS = CGF.EmitReferenceBindingToExpr(MemberInit->getInit(), Field);
|
2009-12-25 06:46:43 +08:00
|
|
|
CGF.EmitStoreThroughLValue(RHS, LHS, FieldType);
|
2010-02-01 03:07:50 +08:00
|
|
|
} else if (FieldType->isArrayType() && !MemberInit->getInit()) {
|
2010-05-23 01:35:42 +08:00
|
|
|
CGF.EmitNullInitialization(LHS.getAddress(), Field->getType());
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
} else if (!CGF.hasAggregateLLVMType(Field->getType())) {
|
2010-06-04 03:58:07 +08:00
|
|
|
RHS = RValue::get(CGF.EmitScalarExpr(MemberInit->getInit()));
|
2009-12-25 06:46:43 +08:00
|
|
|
CGF.EmitStoreThroughLValue(RHS, LHS, FieldType);
|
Rework base and member initialization in constructors, with several
(necessarily simultaneous) changes:
- CXXBaseOrMemberInitializer now contains only a single initializer
rather than a set of initialiation arguments + a constructor. The
single initializer covers all aspects of initialization, including
constructor calls as necessary but also cleanup of temporaries
created by the initializer (which we never handled
before!).
- Rework + simplify code generation for CXXBaseOrMemberInitializers,
since we can now just emit the initializer as an initializer.
- Switched base and member initialization over to the new
initialization code (InitializationSequence), so that it
- Improved diagnostics for the new initialization code when
initializing bases and members, to match the diagnostics produced
by the previous (special-purpose) code.
- Simplify the representation of type-checked constructor initializers in
templates; instead of keeping the fully-type-checked AST, which is
rather hard to undo at template instantiation time, throw away the
type-checked AST and store the raw expressions in the AST. This
simplifies instantiation, but loses a little but of information in
the AST.
- When type-checking implicit base or member initializers within a
dependent context, don't add the generated initializers into the
AST, because they'll look like they were explicit.
- Record in CXXConstructExpr when the constructor call is to
initialize a base class, so that CodeGen does not have to infer it
from context. This ensures that we call the right kind of
constructor.
There are also a few "opportunity" fixes here that were needed to not
regress, for example:
- Diagnose default-initialization of a const-qualified class that
does not have a user-declared default constructor. We had this
diagnostic specifically for bases and members, but missed it for
variables. That's fixed now.
- When defining the implicit constructors, destructor, and
copy-assignment operator, set the CurContext to that constructor
when we're defining the body.
llvm-svn: 94952
2010-01-31 17:12:51 +08:00
|
|
|
} else if (MemberInit->getInit()->getType()->isAnyComplexType()) {
|
|
|
|
CGF.EmitComplexExprIntoAddr(MemberInit->getInit(), LHS.getAddress(),
|
2009-12-25 06:46:43 +08:00
|
|
|
LHS.isVolatileQualified());
|
|
|
|
} else {
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
llvm::Value *ArrayIndexVar = 0;
|
|
|
|
const ConstantArrayType *Array
|
|
|
|
= CGF.getContext().getAsConstantArrayType(FieldType);
|
|
|
|
if (Array && Constructor->isImplicit() &&
|
|
|
|
Constructor->isCopyConstructor()) {
|
|
|
|
const llvm::Type *SizeTy
|
|
|
|
= CGF.ConvertType(CGF.getContext().getSizeType());
|
|
|
|
|
|
|
|
// The LHS is a pointer to the first object we'll be constructing, as
|
|
|
|
// a flat array.
|
|
|
|
QualType BaseElementTy = CGF.getContext().getBaseElementType(Array);
|
|
|
|
const llvm::Type *BasePtr = CGF.ConvertType(BaseElementTy);
|
|
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
|
|
llvm::Value *BaseAddrPtr = CGF.Builder.CreateBitCast(LHS.getAddress(),
|
|
|
|
BasePtr);
|
2010-08-21 11:08:16 +08:00
|
|
|
LHS = CGF.MakeAddrLValue(BaseAddrPtr, BaseElementTy);
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
|
|
|
|
// Create an array index that will be used to walk over all of the
|
|
|
|
// objects we're constructing.
|
|
|
|
ArrayIndexVar = CGF.CreateTempAlloca(SizeTy, "object.index");
|
|
|
|
llvm::Value *Zero = llvm::Constant::getNullValue(SizeTy);
|
|
|
|
CGF.Builder.CreateStore(Zero, ArrayIndexVar);
|
|
|
|
|
|
|
|
// If we are copying an array of scalars or classes with trivial copy
|
|
|
|
// constructors, perform a single aggregate copy.
|
|
|
|
const RecordType *Record = BaseElementTy->getAs<RecordType>();
|
|
|
|
if (!Record ||
|
|
|
|
cast<CXXRecordDecl>(Record->getDecl())->hasTrivialCopyConstructor()) {
|
|
|
|
// Find the source pointer. We knows it's the last argument because
|
|
|
|
// we know we're in a copy constructor.
|
|
|
|
unsigned SrcArgIndex = Args.size() - 1;
|
|
|
|
llvm::Value *SrcPtr
|
2011-03-09 12:27:21 +08:00
|
|
|
= CGF.Builder.CreateLoad(CGF.GetAddrOfLocalVar(Args[SrcArgIndex]));
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
LValue Src = CGF.EmitLValueForFieldInitialization(SrcPtr, Field, 0);
|
|
|
|
|
|
|
|
// Copy the aggregate.
|
|
|
|
CGF.EmitAggregateCopy(LHS.getAddress(), Src.getAddress(), FieldType,
|
|
|
|
LHS.isVolatileQualified());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Emit the block variables for the array indices, if any.
|
|
|
|
for (unsigned I = 0, N = MemberInit->getNumArrayIndices(); I != N; ++I)
|
2010-10-15 12:57:14 +08:00
|
|
|
CGF.EmitAutoVarDecl(*MemberInit->getArrayIndex(I));
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
EmitAggMemberInitializer(CGF, LHS, ArrayIndexVar, MemberInit, FieldType, 0);
|
2010-02-07 03:50:17 +08:00
|
|
|
|
2011-02-28 08:33:03 +08:00
|
|
|
if (!CGF.CGM.getLangOptions().Exceptions)
|
2010-02-07 03:50:17 +08:00
|
|
|
return;
|
|
|
|
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
// FIXME: If we have an array of classes w/ non-trivial destructors,
|
|
|
|
// we need to destroy in reverse order of construction along the exception
|
|
|
|
// path.
|
2010-02-07 03:50:17 +08:00
|
|
|
const RecordType *RT = FieldType->getAs<RecordType>();
|
|
|
|
if (!RT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
CXXRecordDecl *RD = cast<CXXRecordDecl>(RT->getDecl());
|
2010-07-21 09:23:41 +08:00
|
|
|
if (!RD->hasTrivialDestructor())
|
2010-07-21 15:22:38 +08:00
|
|
|
CGF.EHStack.pushCleanup<CallMemberDtor>(EHCleanup, Field,
|
|
|
|
RD->getDestructor());
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
/// Checks whether the given constructor is a valid subject for the
|
|
|
|
/// complete-to-base constructor delegation optimization, i.e.
|
|
|
|
/// emitting the complete constructor as a simple call to the base
|
|
|
|
/// constructor.
|
|
|
|
static bool IsConstructorDelegationValid(const CXXConstructorDecl *Ctor) {
|
|
|
|
|
|
|
|
// Currently we disable the optimization for classes with virtual
|
|
|
|
// bases because (1) the addresses of parameter variables need to be
|
|
|
|
// consistent across all initializers but (2) the delegate function
|
|
|
|
// call necessarily creates a second copy of the parameter variable.
|
|
|
|
//
|
|
|
|
// The limiting example (purely theoretical AFAIK):
|
|
|
|
// struct A { A(int &c) { c++; } };
|
|
|
|
// struct B : virtual A {
|
|
|
|
// B(int count) : A(count) { printf("%d\n", count); }
|
|
|
|
// };
|
|
|
|
// ...although even this example could in principle be emitted as a
|
|
|
|
// delegation since the address of the parameter doesn't escape.
|
|
|
|
if (Ctor->getParent()->getNumVBases()) {
|
|
|
|
// TODO: white-list trivial vbase initializers. This case wouldn't
|
|
|
|
// be subject to the restrictions below.
|
|
|
|
|
|
|
|
// TODO: white-list cases where:
|
|
|
|
// - there are no non-reference parameters to the constructor
|
|
|
|
// - the initializers don't access any non-reference parameters
|
|
|
|
// - the initializers don't take the address of non-reference
|
|
|
|
// parameters
|
|
|
|
// - etc.
|
|
|
|
// If we ever add any of the above cases, remember that:
|
|
|
|
// - function-try-blocks will always blacklist this optimization
|
|
|
|
// - we need to perform the constructor prologue and cleanup in
|
|
|
|
// EmitConstructorBody.
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We also disable the optimization for variadic functions because
|
|
|
|
// it's impossible to "re-pass" varargs.
|
|
|
|
if (Ctor->getType()->getAs<FunctionProtoType>()->isVariadic())
|
|
|
|
return false;
|
|
|
|
|
2011-05-01 15:04:31 +08:00
|
|
|
// FIXME: Decide if we can do a delegation of a delegating constructor.
|
|
|
|
if (Ctor->isDelegatingConstructor())
|
|
|
|
return false;
|
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-02-19 17:25:03 +08:00
|
|
|
/// EmitConstructorBody - Emits the body of the current constructor.
|
|
|
|
void CodeGenFunction::EmitConstructorBody(FunctionArgList &Args) {
|
|
|
|
const CXXConstructorDecl *Ctor = cast<CXXConstructorDecl>(CurGD.getDecl());
|
|
|
|
CXXCtorType CtorType = CurGD.getCtorType();
|
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
// Before we go any further, try the complete->base constructor
|
|
|
|
// delegation optimization.
|
|
|
|
if (CtorType == Ctor_Complete && IsConstructorDelegationValid(Ctor)) {
|
2010-08-12 05:04:37 +08:00
|
|
|
if (CGDebugInfo *DI = getDebugInfo())
|
|
|
|
DI->EmitStopPoint(Builder);
|
2010-02-23 08:48:20 +08:00
|
|
|
EmitDelegateCXXConstructorCall(Ctor, Ctor_Base, Args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-02-19 17:25:03 +08:00
|
|
|
Stmt *Body = Ctor->getBody();
|
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
// Enter the function-try-block before the constructor prologue if
|
|
|
|
// applicable.
|
|
|
|
bool IsTryBody = (Body && isa<CXXTryStmt>(Body));
|
|
|
|
if (IsTryBody)
|
2010-07-07 14:56:46 +08:00
|
|
|
EnterCXXTryStmt(*cast<CXXTryStmt>(Body), true);
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-07-06 09:34:17 +08:00
|
|
|
EHScopeStack::stable_iterator CleanupDepth = EHStack.stable_begin();
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
// Emit the constructor prologue, i.e. the base and member
|
|
|
|
// initializers.
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
EmitCtorPrologue(Ctor, CtorType, Args);
|
2010-02-19 17:25:03 +08:00
|
|
|
|
|
|
|
// Emit the body of the statement.
|
2010-02-23 08:48:20 +08:00
|
|
|
if (IsTryBody)
|
2010-02-19 17:25:03 +08:00
|
|
|
EmitStmt(cast<CXXTryStmt>(Body)->getTryBlock());
|
|
|
|
else if (Body)
|
|
|
|
EmitStmt(Body);
|
|
|
|
|
|
|
|
// Emit any cleanup blocks associated with the member or base
|
|
|
|
// initializers, which includes (along the exceptional path) the
|
|
|
|
// destructors for those members and bases that were fully
|
|
|
|
// constructed.
|
2010-07-06 09:34:17 +08:00
|
|
|
PopCleanupBlocks(CleanupDepth);
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
if (IsTryBody)
|
2010-07-07 14:56:46 +08:00
|
|
|
ExitCXXTryStmt(*cast<CXXTryStmt>(Body), true);
|
2010-02-19 17:25:03 +08:00
|
|
|
}
|
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
/// EmitCtorPrologue - This routine generates necessary code to initialize
|
|
|
|
/// base classes and non-static data members belonging to this constructor.
|
|
|
|
void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD,
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
CXXCtorType CtorType,
|
|
|
|
FunctionArgList &Args) {
|
2011-05-01 15:04:31 +08:00
|
|
|
if (CD->isDelegatingConstructor())
|
|
|
|
return EmitDelegatingCXXConstructorCall(CD, Args);
|
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
const CXXRecordDecl *ClassDecl = CD->getParent();
|
2010-02-03 03:58:43 +08:00
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
llvm::SmallVector<CXXCtorInitializer *, 8> MemberInitializers;
|
2009-12-25 06:46:43 +08:00
|
|
|
|
|
|
|
for (CXXConstructorDecl::init_const_iterator B = CD->init_begin(),
|
|
|
|
E = CD->init_end();
|
|
|
|
B != E; ++B) {
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *Member = (*B);
|
2009-12-25 06:46:43 +08:00
|
|
|
|
2011-05-04 04:19:28 +08:00
|
|
|
if (Member->isBaseInitializer()) {
|
2009-12-25 06:46:43 +08:00
|
|
|
EmitBaseInitializer(*this, ClassDecl, Member, CtorType);
|
2011-05-04 04:19:28 +08:00
|
|
|
} else {
|
|
|
|
assert(Member->isAnyMemberInitializer() &&
|
|
|
|
"Delegating initializer on non-delegating constructor");
|
2010-02-03 03:58:43 +08:00
|
|
|
MemberInitializers.push_back(Member);
|
2011-05-04 04:19:28 +08:00
|
|
|
}
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
InitializeVTablePointers(ClassDecl);
|
2010-02-03 03:58:43 +08:00
|
|
|
|
2010-07-06 09:34:17 +08:00
|
|
|
for (unsigned I = 0, E = MemberInitializers.size(); I != E; ++I)
|
Reimplement code generation for copying fields in the
implicitly-generated copy constructor. Previously, Sema would perform
some checking and instantiation to determine which copy constructors,
etc., would be called, then CodeGen would attempt to figure out which
copy constructor to call... but would get it wrong, or poke at an
uninstantiated default argument, or fail in other ways.
The new scheme is similar to what we now do for the implicit
copy-assignment operator, where Sema performs all of the semantic
analysis and builds specific ASTs that look similar to the ASTs we'd
get from explicitly writing the copy constructor, so that CodeGen need
only do a direct translation.
However, it's not quite that simple because one cannot explicit write
elementwise copy-construction of an array. So, I've extended
CXXBaseOrMemberInitializer to contain a list of indexing variables
used to copy-construct the elements. For example, if we have:
struct A { A(const A&); };
struct B {
A array[2][3];
};
then we generate an implicit copy assignment operator for B that looks
something like this:
B::B(const B &other) : array[i0][i1](other.array[i0][i1]) { }
CodeGen will loop over the invented variables i0 and i1 to visit all
elements in the array, so that each element in the destination array
will be copy-constructed from the corresponding element in the source
array. Of course, if we're dealing with arrays of scalars or class
types with trivial copy-assignment operators, we just generate a
memcpy rather than a loop.
Fixes PR6928, PR5989, and PR6887. Boost.Regex now compiles and passes
all of its regression tests.
Conspicuously missing from this patch is handling for the exceptional
case, where we need to destruct those objects that we have
constructed. I'll address that case separately.
llvm-svn: 103079
2010-05-05 13:51:00 +08:00
|
|
|
EmitMemberInitializer(*this, ClassDecl, MemberInitializers[I], CD, Args);
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
|
2010-02-19 17:25:03 +08:00
|
|
|
/// EmitDestructorBody - Emits the body of the current destructor.
|
|
|
|
void CodeGenFunction::EmitDestructorBody(FunctionArgList &Args) {
|
|
|
|
const CXXDestructorDecl *Dtor = cast<CXXDestructorDecl>(CurGD.getDecl());
|
|
|
|
CXXDtorType DtorType = CurGD.getDtorType();
|
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
// The call to operator delete in a deleting destructor happens
|
|
|
|
// outside of the function-try-block, which means it's always
|
|
|
|
// possible to delegate the destructor body to the complete
|
|
|
|
// destructor. Do so.
|
|
|
|
if (DtorType == Dtor_Deleting) {
|
|
|
|
EnterDtorCleanups(Dtor, Dtor_Deleting);
|
|
|
|
EmitCXXDestructorCall(Dtor, Dtor_Complete, /*ForVirtualBase=*/false,
|
|
|
|
LoadCXXThis());
|
|
|
|
PopCleanupBlock();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-02-19 17:25:03 +08:00
|
|
|
Stmt *Body = Dtor->getBody();
|
|
|
|
|
|
|
|
// If the body is a function-try-block, enter the try before
|
2010-07-21 13:30:47 +08:00
|
|
|
// anything else.
|
|
|
|
bool isTryBody = (Body && isa<CXXTryStmt>(Body));
|
2010-02-19 17:25:03 +08:00
|
|
|
if (isTryBody)
|
2010-07-07 14:56:46 +08:00
|
|
|
EnterCXXTryStmt(*cast<CXXTryStmt>(Body), true);
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
// Enter the epilogue cleanups.
|
|
|
|
RunCleanupsScope DtorEpilogue(*this);
|
|
|
|
|
2010-02-19 17:25:03 +08:00
|
|
|
// If this is the complete variant, just invoke the base variant;
|
|
|
|
// the epilogue will destruct the virtual bases. But we can't do
|
|
|
|
// this optimization if the body is a function-try-block, because
|
|
|
|
// we'd introduce *two* handler blocks.
|
2010-07-21 13:30:47 +08:00
|
|
|
switch (DtorType) {
|
|
|
|
case Dtor_Deleting: llvm_unreachable("already handled deleting case");
|
|
|
|
|
|
|
|
case Dtor_Complete:
|
|
|
|
// Enter the cleanup scopes for virtual bases.
|
|
|
|
EnterDtorCleanups(Dtor, Dtor_Complete);
|
|
|
|
|
|
|
|
if (!isTryBody) {
|
|
|
|
EmitCXXDestructorCall(Dtor, Dtor_Base, /*ForVirtualBase=*/false,
|
|
|
|
LoadCXXThis());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Fallthrough: act like we're in the base variant.
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
case Dtor_Base:
|
|
|
|
// Enter the cleanup scopes for fields and non-virtual bases.
|
|
|
|
EnterDtorCleanups(Dtor, Dtor_Base);
|
|
|
|
|
|
|
|
// Initialize the vtable pointers before entering the body.
|
2010-03-29 05:07:49 +08:00
|
|
|
InitializeVTablePointers(Dtor->getParent());
|
2010-02-19 17:25:03 +08:00
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
if (isTryBody)
|
|
|
|
EmitStmt(cast<CXXTryStmt>(Body)->getTryBlock());
|
|
|
|
else if (Body)
|
|
|
|
EmitStmt(Body);
|
|
|
|
else {
|
|
|
|
assert(Dtor->isImplicit() && "bodyless dtor not implicit");
|
|
|
|
// nothing to do besides what's in the epilogue
|
|
|
|
}
|
2011-02-03 07:12:46 +08:00
|
|
|
// -fapple-kext must inline any call to this dtor into
|
|
|
|
// the caller's body.
|
|
|
|
if (getContext().getLangOptions().AppleKext)
|
|
|
|
CurFn->addFnAttr(llvm::Attribute::AlwaysInline);
|
2010-07-21 13:30:47 +08:00
|
|
|
break;
|
2010-02-19 17:25:03 +08:00
|
|
|
}
|
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
// Jump out through the epilogue cleanups.
|
|
|
|
DtorEpilogue.ForceCleanup();
|
2010-02-19 17:25:03 +08:00
|
|
|
|
|
|
|
// Exit the try if applicable.
|
|
|
|
if (isTryBody)
|
2010-07-07 14:56:46 +08:00
|
|
|
ExitCXXTryStmt(*cast<CXXTryStmt>(Body), true);
|
2010-02-19 17:25:03 +08:00
|
|
|
}
|
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
namespace {
|
|
|
|
/// Call the operator delete associated with the current destructor.
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallDtorDelete : EHScopeStack::Cleanup {
|
2010-07-21 13:30:47 +08:00
|
|
|
CallDtorDelete() {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
const CXXDestructorDecl *Dtor = cast<CXXDestructorDecl>(CGF.CurCodeDecl);
|
|
|
|
const CXXRecordDecl *ClassDecl = Dtor->getParent();
|
|
|
|
CGF.EmitDeleteCall(Dtor->getOperatorDelete(), CGF.LoadCXXThis(),
|
|
|
|
CGF.getContext().getTagDeclType(ClassDecl));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallArrayFieldDtor : EHScopeStack::Cleanup {
|
2010-07-21 13:30:47 +08:00
|
|
|
const FieldDecl *Field;
|
|
|
|
CallArrayFieldDtor(const FieldDecl *Field) : Field(Field) {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
QualType FieldType = Field->getType();
|
|
|
|
const ConstantArrayType *Array =
|
|
|
|
CGF.getContext().getAsConstantArrayType(FieldType);
|
|
|
|
|
|
|
|
QualType BaseType =
|
|
|
|
CGF.getContext().getBaseElementType(Array->getElementType());
|
|
|
|
const CXXRecordDecl *FieldClassDecl = BaseType->getAsCXXRecordDecl();
|
|
|
|
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
|
|
LValue LHS = CGF.EmitLValueForField(ThisPtr, Field,
|
|
|
|
// FIXME: Qualifiers?
|
|
|
|
/*CVRQualifiers=*/0);
|
|
|
|
|
|
|
|
const llvm::Type *BasePtr = CGF.ConvertType(BaseType)->getPointerTo();
|
|
|
|
llvm::Value *BaseAddrPtr =
|
|
|
|
CGF.Builder.CreateBitCast(LHS.getAddress(), BasePtr);
|
|
|
|
CGF.EmitCXXAggrDestructorCall(FieldClassDecl->getDestructor(),
|
|
|
|
Array, BaseAddrPtr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallFieldDtor : EHScopeStack::Cleanup {
|
2010-07-21 13:30:47 +08:00
|
|
|
const FieldDecl *Field;
|
|
|
|
CallFieldDtor(const FieldDecl *Field) : Field(Field) {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
const CXXRecordDecl *FieldClassDecl =
|
|
|
|
Field->getType()->getAsCXXRecordDecl();
|
|
|
|
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
|
|
LValue LHS = CGF.EmitLValueForField(ThisPtr, Field,
|
|
|
|
// FIXME: Qualifiers?
|
|
|
|
/*CVRQualifiers=*/0);
|
|
|
|
|
|
|
|
CGF.EmitCXXDestructorCall(FieldClassDecl->getDestructor(),
|
|
|
|
Dtor_Complete, /*ForVirtualBase=*/false,
|
|
|
|
LHS.getAddress());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2009-12-25 06:46:43 +08:00
|
|
|
/// EmitDtorEpilogue - Emit all code that comes at the end of class's
|
|
|
|
/// destructor. This is to call destructors on members and base classes
|
|
|
|
/// in reverse order of their construction.
|
2010-07-21 13:30:47 +08:00
|
|
|
void CodeGenFunction::EnterDtorCleanups(const CXXDestructorDecl *DD,
|
|
|
|
CXXDtorType DtorType) {
|
2009-12-25 06:46:43 +08:00
|
|
|
assert(!DD->isTrivial() &&
|
|
|
|
"Should not emit dtor epilogue for trivial dtor!");
|
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
// The deleting-destructor phase just needs to call the appropriate
|
|
|
|
// operator delete that Sema picked up.
|
2010-02-19 03:59:28 +08:00
|
|
|
if (DtorType == Dtor_Deleting) {
|
|
|
|
assert(DD->getOperatorDelete() &&
|
|
|
|
"operator delete missing - EmitDtorEpilogue");
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallDtorDelete>(NormalAndEHCleanup);
|
2010-02-19 03:59:28 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
const CXXRecordDecl *ClassDecl = DD->getParent();
|
|
|
|
|
|
|
|
// The complete-destructor phase just destructs all the virtual bases.
|
2010-02-19 03:59:28 +08:00
|
|
|
if (DtorType == Dtor_Complete) {
|
2010-07-21 13:30:47 +08:00
|
|
|
|
|
|
|
// We push them in the forward order so that they'll be popped in
|
|
|
|
// the reverse order.
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator I =
|
|
|
|
ClassDecl->vbases_begin(), E = ClassDecl->vbases_end();
|
2010-02-19 03:59:28 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
const CXXBaseSpecifier &Base = *I;
|
|
|
|
CXXRecordDecl *BaseClassDecl
|
|
|
|
= cast<CXXRecordDecl>(Base.getType()->getAs<RecordType>()->getDecl());
|
|
|
|
|
|
|
|
// Ignore trivial destructors.
|
|
|
|
if (BaseClassDecl->hasTrivialDestructor())
|
|
|
|
continue;
|
2010-07-21 13:30:47 +08:00
|
|
|
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallBaseDtor>(NormalAndEHCleanup,
|
|
|
|
BaseClassDecl,
|
|
|
|
/*BaseIsVirtual*/ true);
|
2010-02-19 03:59:28 +08:00
|
|
|
}
|
2010-07-21 13:30:47 +08:00
|
|
|
|
2010-02-19 03:59:28 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(DtorType == Dtor_Base);
|
2010-07-21 13:30:47 +08:00
|
|
|
|
|
|
|
// Destroy non-virtual bases.
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator I =
|
|
|
|
ClassDecl->bases_begin(), E = ClassDecl->bases_end(); I != E; ++I) {
|
|
|
|
const CXXBaseSpecifier &Base = *I;
|
|
|
|
|
|
|
|
// Ignore virtual bases.
|
|
|
|
if (Base.isVirtual())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CXXRecordDecl *BaseClassDecl = Base.getType()->getAsCXXRecordDecl();
|
|
|
|
|
|
|
|
// Ignore trivial destructors.
|
|
|
|
if (BaseClassDecl->hasTrivialDestructor())
|
|
|
|
continue;
|
2010-02-19 03:59:28 +08:00
|
|
|
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallBaseDtor>(NormalAndEHCleanup,
|
|
|
|
BaseClassDecl,
|
|
|
|
/*BaseIsVirtual*/ false);
|
2010-07-21 13:30:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Destroy direct fields.
|
2009-12-25 06:46:43 +08:00
|
|
|
llvm::SmallVector<const FieldDecl *, 16> FieldDecls;
|
|
|
|
for (CXXRecordDecl::field_iterator I = ClassDecl->field_begin(),
|
|
|
|
E = ClassDecl->field_end(); I != E; ++I) {
|
|
|
|
const FieldDecl *Field = *I;
|
|
|
|
|
|
|
|
QualType FieldType = getContext().getCanonicalType(Field->getType());
|
2010-07-21 13:30:47 +08:00
|
|
|
const ConstantArrayType *Array =
|
|
|
|
getContext().getAsConstantArrayType(FieldType);
|
|
|
|
if (Array)
|
|
|
|
FieldType = getContext().getBaseElementType(Array->getElementType());
|
2009-12-25 06:46:43 +08:00
|
|
|
|
|
|
|
const RecordType *RT = FieldType->getAs<RecordType>();
|
|
|
|
if (!RT)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CXXRecordDecl *FieldClassDecl = cast<CXXRecordDecl>(RT->getDecl());
|
|
|
|
if (FieldClassDecl->hasTrivialDestructor())
|
|
|
|
continue;
|
2010-05-03 07:57:15 +08:00
|
|
|
|
2010-07-21 13:30:47 +08:00
|
|
|
if (Array)
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallArrayFieldDtor>(NormalAndEHCleanup, Field);
|
2010-07-21 13:30:47 +08:00
|
|
|
else
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallFieldDtor>(NormalAndEHCleanup, Field);
|
2009-12-25 06:46:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-02 04:29:01 +08:00
|
|
|
/// EmitCXXAggrConstructorCall - This routine essentially creates a (nested)
|
|
|
|
/// for-loop to call the default constructor on individual members of the
|
|
|
|
/// array.
|
|
|
|
/// 'D' is the default constructor for elements of the array, 'ArrayTy' is the
|
|
|
|
/// array type and 'ArrayPtr' points to the beginning fo the array.
|
|
|
|
/// It is assumed that all relevant checks have been made by the caller.
|
2010-07-21 09:10:17 +08:00
|
|
|
///
|
|
|
|
/// \param ZeroInitialization True if each element should be zero-initialized
|
|
|
|
/// before it is constructed.
|
2010-01-02 04:29:01 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D,
|
2010-07-21 09:10:17 +08:00
|
|
|
const ConstantArrayType *ArrayTy,
|
|
|
|
llvm::Value *ArrayPtr,
|
|
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
|
|
CallExpr::const_arg_iterator ArgEnd,
|
|
|
|
bool ZeroInitialization) {
|
2010-01-02 04:29:01 +08:00
|
|
|
|
|
|
|
const llvm::Type *SizeTy = ConvertType(getContext().getSizeType());
|
|
|
|
llvm::Value * NumElements =
|
|
|
|
llvm::ConstantInt::get(SizeTy,
|
|
|
|
getContext().getConstantArrayElementCount(ArrayTy));
|
|
|
|
|
2010-07-21 09:10:17 +08:00
|
|
|
EmitCXXAggrConstructorCall(D, NumElements, ArrayPtr, ArgBeg, ArgEnd,
|
|
|
|
ZeroInitialization);
|
2010-01-02 04:29:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D,
|
|
|
|
llvm::Value *NumElements,
|
|
|
|
llvm::Value *ArrayPtr,
|
|
|
|
CallExpr::const_arg_iterator ArgBeg,
|
2010-07-21 09:10:17 +08:00
|
|
|
CallExpr::const_arg_iterator ArgEnd,
|
|
|
|
bool ZeroInitialization) {
|
2010-01-02 04:29:01 +08:00
|
|
|
const llvm::Type *SizeTy = ConvertType(getContext().getSizeType());
|
|
|
|
|
|
|
|
// Create a temporary for the loop index and initialize it with 0.
|
|
|
|
llvm::Value *IndexPtr = CreateTempAlloca(SizeTy, "loop.index");
|
|
|
|
llvm::Value *Zero = llvm::Constant::getNullValue(SizeTy);
|
|
|
|
Builder.CreateStore(Zero, IndexPtr);
|
|
|
|
|
|
|
|
// Start the loop with a block that tests the condition.
|
|
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
|
|
|
|
|
|
// Generate: if (loop-index < number-of-elements fall to the loop body,
|
|
|
|
// otherwise, go to the block after the for-loop.
|
|
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElements, "isless");
|
|
|
|
// If the condition is true, execute the body.
|
|
|
|
Builder.CreateCondBr(IsLess, ForBody, AfterFor);
|
|
|
|
|
|
|
|
EmitBlock(ForBody);
|
|
|
|
|
|
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
|
|
// Inside the loop body, emit the constructor call on the array element.
|
|
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
llvm::Value *Address = Builder.CreateInBoundsGEP(ArrayPtr, Counter,
|
|
|
|
"arrayidx");
|
|
|
|
|
2010-07-21 09:10:17 +08:00
|
|
|
// Zero initialize the storage, if requested.
|
|
|
|
if (ZeroInitialization)
|
|
|
|
EmitNullInitialization(Address,
|
|
|
|
getContext().getTypeDeclType(D->getParent()));
|
|
|
|
|
2010-01-02 04:29:01 +08:00
|
|
|
// C++ [class.temporary]p4:
|
|
|
|
// There are two contexts in which temporaries are destroyed at a different
|
|
|
|
// point than the end of the full-expression. The first context is when a
|
|
|
|
// default constructor is called to initialize an element of an array.
|
|
|
|
// If the constructor has one or more default arguments, the destruction of
|
|
|
|
// every temporary created in a default argument expression is sequenced
|
|
|
|
// before the construction of the next array element, if any.
|
|
|
|
|
|
|
|
// Keep track of the current number of live temporaries.
|
2010-03-30 11:14:41 +08:00
|
|
|
{
|
2010-07-06 09:34:17 +08:00
|
|
|
RunCleanupsScope Scope(*this);
|
2010-01-02 04:29:01 +08:00
|
|
|
|
2010-05-03 07:20:53 +08:00
|
|
|
EmitCXXConstructorCall(D, Ctor_Complete, /*ForVirtualBase=*/false, Address,
|
2010-05-03 07:01:10 +08:00
|
|
|
ArgBeg, ArgEnd);
|
2010-03-30 11:14:41 +08:00
|
|
|
}
|
2010-01-02 04:29:01 +08:00
|
|
|
|
|
|
|
EmitBlock(ContinueBlock);
|
|
|
|
|
|
|
|
// Emit the increment of the loop counter.
|
|
|
|
llvm::Value *NextVal = llvm::ConstantInt::get(SizeTy, 1);
|
|
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
NextVal = Builder.CreateAdd(Counter, NextVal, "inc");
|
|
|
|
Builder.CreateStore(NextVal, IndexPtr);
|
|
|
|
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
|
|
EmitBranch(CondBlock);
|
|
|
|
|
|
|
|
// Emit the fall-through block.
|
|
|
|
EmitBlock(AfterFor, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// EmitCXXAggrDestructorCall - calls the default destructor on array
|
|
|
|
/// elements in reverse order of construction.
|
|
|
|
void
|
|
|
|
CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D,
|
|
|
|
const ArrayType *Array,
|
|
|
|
llvm::Value *This) {
|
|
|
|
const ConstantArrayType *CA = dyn_cast<ConstantArrayType>(Array);
|
|
|
|
assert(CA && "Do we support VLA for destruction ?");
|
|
|
|
uint64_t ElementCount = getContext().getConstantArrayElementCount(CA);
|
|
|
|
|
|
|
|
const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType());
|
|
|
|
llvm::Value* ElementCountPtr = llvm::ConstantInt::get(SizeLTy, ElementCount);
|
|
|
|
EmitCXXAggrDestructorCall(D, ElementCountPtr, This);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// EmitCXXAggrDestructorCall - calls the default destructor on array
|
|
|
|
/// elements in reverse order of construction.
|
|
|
|
void
|
|
|
|
CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D,
|
|
|
|
llvm::Value *UpperCount,
|
|
|
|
llvm::Value *This) {
|
|
|
|
const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType());
|
|
|
|
llvm::Value *One = llvm::ConstantInt::get(SizeLTy, 1);
|
|
|
|
|
|
|
|
// Create a temporary for the loop index and initialize it with count of
|
|
|
|
// array elements.
|
|
|
|
llvm::Value *IndexPtr = CreateTempAlloca(SizeLTy, "loop.index");
|
|
|
|
|
|
|
|
// Store the number of elements in the index pointer.
|
|
|
|
Builder.CreateStore(UpperCount, IndexPtr);
|
|
|
|
|
|
|
|
// Start the loop with a block that tests the condition.
|
|
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
|
|
|
|
|
|
// Generate: if (loop-index != 0 fall to the loop body,
|
|
|
|
// otherwise, go to the block after the for-loop.
|
|
|
|
llvm::Value* zeroConstant =
|
|
|
|
llvm::Constant::getNullValue(SizeLTy);
|
|
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
llvm::Value *IsNE = Builder.CreateICmpNE(Counter, zeroConstant,
|
|
|
|
"isne");
|
|
|
|
// If the condition is true, execute the body.
|
|
|
|
Builder.CreateCondBr(IsNE, ForBody, AfterFor);
|
|
|
|
|
|
|
|
EmitBlock(ForBody);
|
|
|
|
|
|
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
|
|
// Inside the loop body, emit the constructor call on the array element.
|
|
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
Counter = Builder.CreateSub(Counter, One);
|
|
|
|
llvm::Value *Address = Builder.CreateInBoundsGEP(This, Counter, "arrayidx");
|
2010-05-03 07:29:11 +08:00
|
|
|
EmitCXXDestructorCall(D, Dtor_Complete, /*ForVirtualBase=*/false, Address);
|
2010-01-02 04:29:01 +08:00
|
|
|
|
|
|
|
EmitBlock(ContinueBlock);
|
|
|
|
|
|
|
|
// Emit the decrement of the loop counter.
|
|
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
|
|
Counter = Builder.CreateSub(Counter, One, "dec");
|
|
|
|
Builder.CreateStore(Counter, IndexPtr);
|
|
|
|
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
|
|
EmitBranch(CondBlock);
|
|
|
|
|
|
|
|
// Emit the fall-through block.
|
|
|
|
EmitBlock(AfterFor, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CodeGenFunction::EmitCXXConstructorCall(const CXXConstructorDecl *D,
|
2010-05-03 07:20:53 +08:00
|
|
|
CXXCtorType Type, bool ForVirtualBase,
|
2010-01-02 04:29:01 +08:00
|
|
|
llvm::Value *This,
|
|
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
2011-02-23 04:55:26 +08:00
|
|
|
|
|
|
|
CGDebugInfo *DI = getDebugInfo();
|
|
|
|
if (DI && CGM.getCodeGenOpts().LimitDebugInfo) {
|
|
|
|
// If debug info for this class has been emitted then this is the right time
|
|
|
|
// to do so.
|
|
|
|
const CXXRecordDecl *Parent = D->getParent();
|
|
|
|
DI->getOrCreateRecordType(CGM.getContext().getTypeDeclType(Parent),
|
|
|
|
Parent->getLocation());
|
|
|
|
}
|
|
|
|
|
2010-02-06 08:25:16 +08:00
|
|
|
if (D->isTrivial()) {
|
|
|
|
if (ArgBeg == ArgEnd) {
|
|
|
|
// Trivial default constructor, no codegen required.
|
|
|
|
assert(D->isDefaultConstructor() &&
|
|
|
|
"trivial 0-arg ctor not a default ctor");
|
2010-01-02 04:29:01 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-02-06 08:25:16 +08:00
|
|
|
|
|
|
|
assert(ArgBeg + 1 == ArgEnd && "unexpected argcount for trivial ctor");
|
|
|
|
assert(D->isCopyConstructor() && "trivial 1-arg ctor not a copy ctor");
|
|
|
|
|
|
|
|
const Expr *E = (*ArgBeg);
|
|
|
|
QualType Ty = E->getType();
|
|
|
|
llvm::Value *Src = EmitLValue(E).getAddress();
|
|
|
|
EmitAggregateCopy(This, Src, Ty);
|
2010-01-02 04:29:01 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-05-03 07:33:10 +08:00
|
|
|
llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(D, Type), ForVirtualBase);
|
2010-01-02 04:29:01 +08:00
|
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(D, Type);
|
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
EmitCXXMemberCall(D, Callee, ReturnValueSlot(), This, VTT, ArgBeg, ArgEnd);
|
2010-01-02 04:29:01 +08:00
|
|
|
}
|
|
|
|
|
2010-11-14 05:53:34 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::EmitSynthesizedCXXCopyCtorCall(const CXXConstructorDecl *D,
|
|
|
|
llvm::Value *This, llvm::Value *Src,
|
|
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
|
|
|
if (D->isTrivial()) {
|
|
|
|
assert(ArgBeg + 1 == ArgEnd && "unexpected argcount for trivial ctor");
|
|
|
|
assert(D->isCopyConstructor() && "trivial 1-arg ctor not a copy ctor");
|
|
|
|
EmitAggregateCopy(This, Src, (*ArgBeg)->getType());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(D,
|
|
|
|
clang::Ctor_Complete);
|
|
|
|
assert(D->isInstance() &&
|
|
|
|
"Trying to emit a member call expr on a static method!");
|
|
|
|
|
|
|
|
const FunctionProtoType *FPT = D->getType()->getAs<FunctionProtoType>();
|
|
|
|
|
|
|
|
CallArgList Args;
|
|
|
|
|
|
|
|
// Push the this ptr.
|
2011-05-03 01:57:46 +08:00
|
|
|
Args.add(RValue::get(This), D->getThisType(getContext()));
|
2010-11-14 05:53:34 +08:00
|
|
|
|
|
|
|
|
|
|
|
// Push the src ptr.
|
|
|
|
QualType QT = *(FPT->arg_type_begin());
|
|
|
|
const llvm::Type *t = CGM.getTypes().ConvertType(QT);
|
|
|
|
Src = Builder.CreateBitCast(Src, t);
|
2011-05-03 01:57:46 +08:00
|
|
|
Args.add(RValue::get(Src), QT);
|
2010-11-14 05:53:34 +08:00
|
|
|
|
|
|
|
// Skip over first argument (Src).
|
|
|
|
++ArgBeg;
|
|
|
|
CallExpr::const_arg_iterator Arg = ArgBeg;
|
|
|
|
for (FunctionProtoType::arg_type_iterator I = FPT->arg_type_begin()+1,
|
|
|
|
E = FPT->arg_type_end(); I != E; ++I, ++Arg) {
|
|
|
|
assert(Arg != ArgEnd && "Running over edge of argument list!");
|
2011-03-12 04:59:21 +08:00
|
|
|
EmitCallArg(Args, *Arg, *I);
|
2010-11-14 05:53:34 +08:00
|
|
|
}
|
|
|
|
// Either we've emitted all the call args, or we have a call to a
|
|
|
|
// variadic function.
|
|
|
|
assert((Arg == ArgEnd || FPT->isVariadic()) &&
|
|
|
|
"Extra arguments in non-variadic function!");
|
|
|
|
// If we still have any arguments, emit them using the type of the argument.
|
|
|
|
for (; Arg != ArgEnd; ++Arg) {
|
|
|
|
QualType ArgType = Arg->getType();
|
2011-03-12 04:59:21 +08:00
|
|
|
EmitCallArg(Args, *Arg, ArgType);
|
2010-11-14 05:53:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
QualType ResultType = FPT->getResultType();
|
|
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, Args,
|
|
|
|
FPT->getExtInfo()),
|
|
|
|
Callee, ReturnValueSlot(), Args, D);
|
|
|
|
}
|
|
|
|
|
2010-02-23 08:48:20 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::EmitDelegateCXXConstructorCall(const CXXConstructorDecl *Ctor,
|
|
|
|
CXXCtorType CtorType,
|
|
|
|
const FunctionArgList &Args) {
|
|
|
|
CallArgList DelegateArgs;
|
|
|
|
|
|
|
|
FunctionArgList::const_iterator I = Args.begin(), E = Args.end();
|
|
|
|
assert(I != E && "no parameters to constructor");
|
|
|
|
|
|
|
|
// this
|
2011-05-03 01:57:46 +08:00
|
|
|
DelegateArgs.add(RValue::get(LoadCXXThis()), (*I)->getType());
|
2010-02-23 08:48:20 +08:00
|
|
|
++I;
|
|
|
|
|
|
|
|
// vtt
|
2010-05-03 07:33:10 +08:00
|
|
|
if (llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(Ctor, CtorType),
|
|
|
|
/*ForVirtualBase=*/false)) {
|
2010-02-23 08:48:20 +08:00
|
|
|
QualType VoidPP = getContext().getPointerType(getContext().VoidPtrTy);
|
2011-05-03 01:57:46 +08:00
|
|
|
DelegateArgs.add(RValue::get(VTT), VoidPP);
|
2010-02-23 08:48:20 +08:00
|
|
|
|
2010-03-23 12:11:45 +08:00
|
|
|
if (CodeGenVTables::needsVTTParameter(CurGD)) {
|
2010-02-23 08:48:20 +08:00
|
|
|
assert(I != E && "cannot skip vtt parameter, already done with args");
|
2011-03-09 12:27:21 +08:00
|
|
|
assert((*I)->getType() == VoidPP && "skipping parameter not of vtt type");
|
2010-02-23 08:48:20 +08:00
|
|
|
++I;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Explicit arguments.
|
|
|
|
for (; I != E; ++I) {
|
2011-03-12 04:59:21 +08:00
|
|
|
const VarDecl *param = *I;
|
|
|
|
EmitDelegateCallArg(DelegateArgs, param);
|
2010-02-23 08:48:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
EmitCall(CGM.getTypes().getFunctionInfo(Ctor, CtorType),
|
|
|
|
CGM.GetAddrOfCXXConstructor(Ctor, CtorType),
|
|
|
|
ReturnValueSlot(), DelegateArgs, Ctor);
|
|
|
|
}
|
|
|
|
|
2011-05-04 07:05:34 +08:00
|
|
|
namespace {
|
|
|
|
struct CallDelegatingCtorDtor : EHScopeStack::Cleanup {
|
|
|
|
const CXXDestructorDecl *Dtor;
|
|
|
|
llvm::Value *Addr;
|
|
|
|
CXXDtorType Type;
|
|
|
|
|
|
|
|
CallDelegatingCtorDtor(const CXXDestructorDecl *D, llvm::Value *Addr,
|
|
|
|
CXXDtorType Type)
|
|
|
|
: Dtor(D), Addr(Addr), Type(Type) {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
CGF.EmitCXXDestructorCall(Dtor, Type, /*ForVirtualBase=*/false,
|
|
|
|
Addr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2011-05-01 15:04:31 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::EmitDelegatingCXXConstructorCall(const CXXConstructorDecl *Ctor,
|
|
|
|
const FunctionArgList &Args) {
|
|
|
|
assert(Ctor->isDelegatingConstructor());
|
|
|
|
|
|
|
|
llvm::Value *ThisPtr = LoadCXXThis();
|
|
|
|
|
|
|
|
AggValueSlot AggSlot = AggValueSlot::forAddr(ThisPtr, false, /*Lifetime*/ true);
|
|
|
|
|
|
|
|
EmitAggExpr(Ctor->init_begin()[0]->getInit(), AggSlot);
|
|
|
|
|
2011-05-04 07:05:34 +08:00
|
|
|
const CXXRecordDecl *ClassDecl = Ctor->getParent();
|
|
|
|
if (CGM.getLangOptions().Exceptions && !ClassDecl->hasTrivialDestructor()) {
|
|
|
|
CXXDtorType Type =
|
|
|
|
CurGD.getCtorType() == Ctor_Complete ? Dtor_Complete : Dtor_Base;
|
|
|
|
|
|
|
|
EHStack.pushCleanup<CallDelegatingCtorDtor>(EHCleanup,
|
|
|
|
ClassDecl->getDestructor(),
|
|
|
|
ThisPtr, Type);
|
|
|
|
}
|
|
|
|
}
|
2011-05-01 15:04:31 +08:00
|
|
|
|
2010-01-02 04:29:01 +08:00
|
|
|
void CodeGenFunction::EmitCXXDestructorCall(const CXXDestructorDecl *DD,
|
|
|
|
CXXDtorType Type,
|
2010-05-03 07:29:11 +08:00
|
|
|
bool ForVirtualBase,
|
2010-01-02 04:29:01 +08:00
|
|
|
llvm::Value *This) {
|
2010-05-03 07:33:10 +08:00
|
|
|
llvm::Value *VTT = GetVTTParameter(*this, GlobalDecl(DD, Type),
|
|
|
|
ForVirtualBase);
|
2011-02-02 07:22:34 +08:00
|
|
|
llvm::Value *Callee = 0;
|
|
|
|
if (getContext().getLangOptions().AppleKext)
|
2011-02-04 03:27:17 +08:00
|
|
|
Callee = BuildAppleKextVirtualDestructorCall(DD, Type,
|
|
|
|
DD->getParent());
|
2011-02-02 07:22:34 +08:00
|
|
|
|
|
|
|
if (!Callee)
|
|
|
|
Callee = CGM.GetAddrOfCXXDestructor(DD, Type);
|
2010-01-02 04:29:01 +08:00
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
EmitCXXMemberCall(DD, Callee, ReturnValueSlot(), This, VTT, 0, 0);
|
2010-01-02 04:29:01 +08:00
|
|
|
}
|
|
|
|
|
2010-07-21 09:41:18 +08:00
|
|
|
namespace {
|
2010-07-21 15:22:38 +08:00
|
|
|
struct CallLocalDtor : EHScopeStack::Cleanup {
|
2010-07-21 09:41:18 +08:00
|
|
|
const CXXDestructorDecl *Dtor;
|
|
|
|
llvm::Value *Addr;
|
|
|
|
|
|
|
|
CallLocalDtor(const CXXDestructorDecl *D, llvm::Value *Addr)
|
|
|
|
: Dtor(D), Addr(Addr) {}
|
|
|
|
|
|
|
|
void Emit(CodeGenFunction &CGF, bool IsForEH) {
|
|
|
|
CGF.EmitCXXDestructorCall(Dtor, Dtor_Complete,
|
|
|
|
/*ForVirtualBase=*/false, Addr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2010-07-21 14:29:51 +08:00
|
|
|
void CodeGenFunction::PushDestructorCleanup(const CXXDestructorDecl *D,
|
|
|
|
llvm::Value *Addr) {
|
2010-07-21 15:22:38 +08:00
|
|
|
EHStack.pushCleanup<CallLocalDtor>(NormalAndEHCleanup, D, Addr);
|
2010-07-21 14:29:51 +08:00
|
|
|
}
|
|
|
|
|
2010-07-06 09:34:17 +08:00
|
|
|
void CodeGenFunction::PushDestructorCleanup(QualType T, llvm::Value *Addr) {
|
|
|
|
CXXRecordDecl *ClassDecl = T->getAsCXXRecordDecl();
|
|
|
|
if (!ClassDecl) return;
|
|
|
|
if (ClassDecl->hasTrivialDestructor()) return;
|
|
|
|
|
|
|
|
const CXXDestructorDecl *D = ClassDecl->getDestructor();
|
2011-04-28 10:15:35 +08:00
|
|
|
assert(D && D->isUsed() && "destructor not marked as used!");
|
2010-07-21 14:29:51 +08:00
|
|
|
PushDestructorCleanup(D, Addr);
|
2010-07-06 09:34:17 +08:00
|
|
|
}
|
|
|
|
|
2010-01-02 04:29:01 +08:00
|
|
|
llvm::Value *
|
2010-01-31 09:36:53 +08:00
|
|
|
CodeGenFunction::GetVirtualBaseClassOffset(llvm::Value *This,
|
|
|
|
const CXXRecordDecl *ClassDecl,
|
2010-01-02 04:29:01 +08:00
|
|
|
const CXXRecordDecl *BaseClassDecl) {
|
2010-10-27 02:44:08 +08:00
|
|
|
llvm::Value *VTablePtr = GetVTablePtr(This, Int8PtrTy);
|
2011-04-07 20:37:09 +08:00
|
|
|
CharUnits VBaseOffsetOffset =
|
2010-03-23 12:11:45 +08:00
|
|
|
CGM.getVTables().getVirtualBaseOffsetOffset(ClassDecl, BaseClassDecl);
|
2010-01-02 04:29:01 +08:00
|
|
|
|
|
|
|
llvm::Value *VBaseOffsetPtr =
|
2011-04-07 20:37:09 +08:00
|
|
|
Builder.CreateConstGEP1_64(VTablePtr, VBaseOffsetOffset.getQuantity(),
|
|
|
|
"vbase.offset.ptr");
|
2010-01-02 04:29:01 +08:00
|
|
|
const llvm::Type *PtrDiffTy =
|
|
|
|
ConvertType(getContext().getPointerDiffType());
|
|
|
|
|
|
|
|
VBaseOffsetPtr = Builder.CreateBitCast(VBaseOffsetPtr,
|
|
|
|
PtrDiffTy->getPointerTo());
|
|
|
|
|
|
|
|
llvm::Value *VBaseOffset = Builder.CreateLoad(VBaseOffsetPtr, "vbase.offset");
|
|
|
|
|
|
|
|
return VBaseOffset;
|
|
|
|
}
|
|
|
|
|
2010-03-29 03:40:00 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::InitializeVTablePointer(BaseSubobject Base,
|
2010-04-20 13:22:15 +08:00
|
|
|
const CXXRecordDecl *NearestVBase,
|
2011-03-23 09:04:18 +08:00
|
|
|
CharUnits OffsetFromNearestVBase,
|
2010-03-29 03:40:00 +08:00
|
|
|
llvm::Constant *VTable,
|
|
|
|
const CXXRecordDecl *VTableClass) {
|
2010-03-29 09:08:49 +08:00
|
|
|
const CXXRecordDecl *RD = Base.getBase();
|
|
|
|
|
2010-03-29 03:40:00 +08:00
|
|
|
// Compute the address point.
|
2010-03-29 09:08:49 +08:00
|
|
|
llvm::Value *VTableAddressPoint;
|
2010-03-29 10:38:51 +08:00
|
|
|
|
2010-03-29 09:08:49 +08:00
|
|
|
// Check if we need to use a vtable from the VTT.
|
2010-03-29 10:38:51 +08:00
|
|
|
if (CodeGenVTables::needsVTTParameter(CurGD) &&
|
2010-04-20 13:22:15 +08:00
|
|
|
(RD->getNumVBases() || NearestVBase)) {
|
2010-03-29 09:08:49 +08:00
|
|
|
// Get the secondary vpointer index.
|
|
|
|
uint64_t VirtualPointerIndex =
|
|
|
|
CGM.getVTables().getSecondaryVirtualPointerIndex(VTableClass, Base);
|
|
|
|
|
|
|
|
/// Load the VTT.
|
|
|
|
llvm::Value *VTT = LoadCXXVTT();
|
|
|
|
if (VirtualPointerIndex)
|
|
|
|
VTT = Builder.CreateConstInBoundsGEP1_64(VTT, VirtualPointerIndex);
|
|
|
|
|
|
|
|
// And load the address point from the VTT.
|
|
|
|
VTableAddressPoint = Builder.CreateLoad(VTT);
|
|
|
|
} else {
|
2010-03-29 10:08:26 +08:00
|
|
|
uint64_t AddressPoint = CGM.getVTables().getAddressPoint(Base, VTableClass);
|
2010-03-29 09:08:49 +08:00
|
|
|
VTableAddressPoint =
|
2010-03-29 03:40:00 +08:00
|
|
|
Builder.CreateConstInBoundsGEP2_64(VTable, 0, AddressPoint);
|
2010-03-29 09:08:49 +08:00
|
|
|
}
|
2010-03-29 03:40:00 +08:00
|
|
|
|
|
|
|
// Compute where to store the address point.
|
2010-05-03 08:29:58 +08:00
|
|
|
llvm::Value *VirtualOffset = 0;
|
2011-03-23 08:45:26 +08:00
|
|
|
CharUnits NonVirtualOffset = CharUnits::Zero();
|
2010-04-21 02:05:10 +08:00
|
|
|
|
|
|
|
if (CodeGenVTables::needsVTTParameter(CurGD) && NearestVBase) {
|
|
|
|
// We need to use the virtual base offset offset because the virtual base
|
|
|
|
// might have a different offset in the most derived class.
|
2010-05-03 08:29:58 +08:00
|
|
|
VirtualOffset = GetVirtualBaseClassOffset(LoadCXXThis(), VTableClass,
|
|
|
|
NearestVBase);
|
2011-03-23 09:04:18 +08:00
|
|
|
NonVirtualOffset = OffsetFromNearestVBase;
|
2010-04-21 02:05:10 +08:00
|
|
|
} else {
|
2010-05-03 08:29:58 +08:00
|
|
|
// We can just use the base offset in the complete class.
|
2011-03-24 09:21:01 +08:00
|
|
|
NonVirtualOffset = Base.getBaseOffset();
|
2010-04-21 02:05:10 +08:00
|
|
|
}
|
2010-05-03 08:29:58 +08:00
|
|
|
|
|
|
|
// Apply the offsets.
|
|
|
|
llvm::Value *VTableField = LoadCXXThis();
|
|
|
|
|
2011-03-23 08:45:26 +08:00
|
|
|
if (!NonVirtualOffset.isZero() || VirtualOffset)
|
2010-05-03 08:29:58 +08:00
|
|
|
VTableField = ApplyNonVirtualAndVirtualOffset(*this, VTableField,
|
|
|
|
NonVirtualOffset,
|
|
|
|
VirtualOffset);
|
2010-04-21 00:22:16 +08:00
|
|
|
|
2010-03-29 03:40:00 +08:00
|
|
|
// Finally, store the address point.
|
|
|
|
const llvm::Type *AddressPointPtrTy =
|
|
|
|
VTableAddressPoint->getType()->getPointerTo();
|
|
|
|
VTableField = Builder.CreateBitCast(VTableField, AddressPointPtrTy);
|
|
|
|
Builder.CreateStore(VTableAddressPoint, VTableField);
|
|
|
|
}
|
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
void
|
|
|
|
CodeGenFunction::InitializeVTablePointers(BaseSubobject Base,
|
2010-04-20 13:22:15 +08:00
|
|
|
const CXXRecordDecl *NearestVBase,
|
2011-03-23 09:04:18 +08:00
|
|
|
CharUnits OffsetFromNearestVBase,
|
2010-03-29 05:07:49 +08:00
|
|
|
bool BaseIsNonVirtualPrimaryBase,
|
|
|
|
llvm::Constant *VTable,
|
|
|
|
const CXXRecordDecl *VTableClass,
|
|
|
|
VisitedVirtualBasesSetTy& VBases) {
|
|
|
|
// If this base is a non-virtual primary base the address point has already
|
|
|
|
// been set.
|
|
|
|
if (!BaseIsNonVirtualPrimaryBase) {
|
|
|
|
// Initialize the vtable pointer for this base.
|
2010-05-03 08:07:07 +08:00
|
|
|
InitializeVTablePointer(Base, NearestVBase, OffsetFromNearestVBase,
|
|
|
|
VTable, VTableClass);
|
2010-03-29 05:07:49 +08:00
|
|
|
}
|
2010-03-29 04:55:21 +08:00
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
const CXXRecordDecl *RD = Base.getBase();
|
2010-03-29 04:55:21 +08:00
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
// Traverse bases.
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
|
|
|
|
E = RD->bases_end(); I != E; ++I) {
|
2010-03-26 12:39:42 +08:00
|
|
|
CXXRecordDecl *BaseDecl
|
|
|
|
= cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
|
2010-03-29 05:07:49 +08:00
|
|
|
|
|
|
|
// Ignore classes without a vtable.
|
|
|
|
if (!BaseDecl->isDynamicClass())
|
|
|
|
continue;
|
|
|
|
|
2011-03-23 09:04:18 +08:00
|
|
|
CharUnits BaseOffset;
|
|
|
|
CharUnits BaseOffsetFromNearestVBase;
|
2010-03-29 09:16:41 +08:00
|
|
|
bool BaseDeclIsNonVirtualPrimaryBase;
|
2010-03-29 05:07:49 +08:00
|
|
|
|
|
|
|
if (I->isVirtual()) {
|
|
|
|
// Check if we've visited this virtual base before.
|
|
|
|
if (!VBases.insert(BaseDecl))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const ASTRecordLayout &Layout =
|
|
|
|
getContext().getASTRecordLayout(VTableClass);
|
|
|
|
|
2011-03-23 09:04:18 +08:00
|
|
|
BaseOffset = Layout.getVBaseClassOffset(BaseDecl);
|
|
|
|
BaseOffsetFromNearestVBase = CharUnits::Zero();
|
2010-03-29 09:16:41 +08:00
|
|
|
BaseDeclIsNonVirtualPrimaryBase = false;
|
2010-03-29 05:07:49 +08:00
|
|
|
} else {
|
|
|
|
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(RD);
|
|
|
|
|
2011-03-24 09:21:01 +08:00
|
|
|
BaseOffset = Base.getBaseOffset() + Layout.getBaseClassOffset(BaseDecl);
|
2010-05-03 08:07:07 +08:00
|
|
|
BaseOffsetFromNearestVBase =
|
2011-03-23 09:04:18 +08:00
|
|
|
OffsetFromNearestVBase + Layout.getBaseClassOffset(BaseDecl);
|
2010-03-29 09:16:41 +08:00
|
|
|
BaseDeclIsNonVirtualPrimaryBase = Layout.getPrimaryBase() == BaseDecl;
|
2010-03-29 05:07:49 +08:00
|
|
|
}
|
2010-03-26 12:39:42 +08:00
|
|
|
|
2011-03-24 09:21:01 +08:00
|
|
|
InitializeVTablePointers(BaseSubobject(BaseDecl, BaseOffset),
|
2010-04-20 13:22:15 +08:00
|
|
|
I->isVirtual() ? BaseDecl : NearestVBase,
|
2010-05-03 08:07:07 +08:00
|
|
|
BaseOffsetFromNearestVBase,
|
2010-03-29 09:16:41 +08:00
|
|
|
BaseDeclIsNonVirtualPrimaryBase,
|
2010-03-29 05:07:49 +08:00
|
|
|
VTable, VTableClass, VBases);
|
2010-01-02 04:29:01 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
void CodeGenFunction::InitializeVTablePointers(const CXXRecordDecl *RD) {
|
|
|
|
// Ignore classes without a vtable.
|
2010-03-26 12:39:42 +08:00
|
|
|
if (!RD->isDynamicClass())
|
2010-01-02 04:29:01 +08:00
|
|
|
return;
|
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
// Get the VTable.
|
|
|
|
llvm::Constant *VTable = CGM.getVTables().GetAddrOfVTable(RD);
|
2010-01-02 04:29:01 +08:00
|
|
|
|
2010-03-29 05:07:49 +08:00
|
|
|
// Initialize the vtable pointers for this class and all of its bases.
|
|
|
|
VisitedVirtualBasesSetTy VBases;
|
2011-03-24 09:21:01 +08:00
|
|
|
InitializeVTablePointers(BaseSubobject(RD, CharUnits::Zero()),
|
|
|
|
/*NearestVBase=*/0,
|
2011-03-23 09:04:18 +08:00
|
|
|
/*OffsetFromNearestVBase=*/CharUnits::Zero(),
|
2010-03-29 05:07:49 +08:00
|
|
|
/*BaseIsNonVirtualPrimaryBase=*/false,
|
|
|
|
VTable, RD, VBases);
|
2010-01-02 04:29:01 +08:00
|
|
|
}
|
2010-10-27 02:44:08 +08:00
|
|
|
|
|
|
|
llvm::Value *CodeGenFunction::GetVTablePtr(llvm::Value *This,
|
|
|
|
const llvm::Type *Ty) {
|
|
|
|
llvm::Value *VTablePtrSrc = Builder.CreateBitCast(This, Ty->getPointerTo());
|
|
|
|
return Builder.CreateLoad(VTablePtrSrc, "vtable");
|
|
|
|
}
|
2011-05-09 04:32:23 +08:00
|
|
|
|
|
|
|
static const CXXRecordDecl *getMostDerivedClassDecl(const Expr *Base) {
|
|
|
|
const Expr *E = Base;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
E = E->IgnoreParens();
|
|
|
|
if (const CastExpr *CE = dyn_cast<CastExpr>(E)) {
|
|
|
|
if (CE->getCastKind() == CK_DerivedToBase ||
|
|
|
|
CE->getCastKind() == CK_UncheckedDerivedToBase ||
|
|
|
|
CE->getCastKind() == CK_NoOp) {
|
|
|
|
E = CE->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType DerivedType = E->getType();
|
|
|
|
if (const PointerType *PTy = DerivedType->getAs<PointerType>())
|
|
|
|
DerivedType = PTy->getPointeeType();
|
|
|
|
|
|
|
|
return cast<CXXRecordDecl>(DerivedType->castAs<RecordType>()->getDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Ideally Expr::IgnoreParenNoopCasts should do this, but it doesn't do
|
|
|
|
// quite what we want.
|
|
|
|
static const Expr *skipNoOpCastsAndParens(const Expr *E) {
|
|
|
|
while (true) {
|
|
|
|
if (const ParenExpr *PE = dyn_cast<ParenExpr>(E)) {
|
|
|
|
E = PE->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const CastExpr *CE = dyn_cast<CastExpr>(E)) {
|
|
|
|
if (CE->getCastKind() == CK_NoOp) {
|
|
|
|
E = CE->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (const UnaryOperator *UO = dyn_cast<UnaryOperator>(E)) {
|
|
|
|
if (UO->getOpcode() == UO_Extension) {
|
|
|
|
E = UO->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return E;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// canDevirtualizeMemberFunctionCall - Checks whether the given virtual member
|
|
|
|
/// function call on the given expr can be devirtualized.
|
|
|
|
/// expr can be devirtualized.
|
|
|
|
static bool canDevirtualizeMemberFunctionCall(const Expr *Base,
|
|
|
|
const CXXMethodDecl *MD) {
|
|
|
|
// If the most derived class is marked final, we know that no subclass can
|
|
|
|
// override this member function and so we can devirtualize it. For example:
|
|
|
|
//
|
|
|
|
// struct A { virtual void f(); }
|
|
|
|
// struct B final : A { };
|
|
|
|
//
|
|
|
|
// void f(B *b) {
|
|
|
|
// b->f();
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
const CXXRecordDecl *MostDerivedClassDecl = getMostDerivedClassDecl(Base);
|
|
|
|
if (MostDerivedClassDecl->hasAttr<FinalAttr>())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// If the member function is marked 'final', we know that it can't be
|
|
|
|
// overridden and can therefore devirtualize it.
|
|
|
|
if (MD->hasAttr<FinalAttr>())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Similarly, if the class itself is marked 'final' it can't be overridden
|
|
|
|
// and we can therefore devirtualize the member function call.
|
|
|
|
if (MD->getParent()->hasAttr<FinalAttr>())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
Base = skipNoOpCastsAndParens(Base);
|
|
|
|
if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Base)) {
|
|
|
|
if (const VarDecl *VD = dyn_cast<VarDecl>(DRE->getDecl())) {
|
|
|
|
// This is a record decl. We know the type and can devirtualize it.
|
|
|
|
return VD->getType()->isRecordType();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can always devirtualize calls on temporary object expressions.
|
|
|
|
if (isa<CXXConstructExpr>(Base))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// And calls on bound temporaries.
|
|
|
|
if (isa<CXXBindTemporaryExpr>(Base))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Check if this is a call expr that returns a record type.
|
|
|
|
if (const CallExpr *CE = dyn_cast<CallExpr>(Base))
|
|
|
|
return CE->getCallReturnType()->isRecordType();
|
|
|
|
|
|
|
|
// We can't devirtualize the call.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool UseVirtualCall(ASTContext &Context,
|
|
|
|
const CXXOperatorCallExpr *CE,
|
|
|
|
const CXXMethodDecl *MD) {
|
|
|
|
if (!MD->isVirtual())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// When building with -fapple-kext, all calls must go through the vtable since
|
|
|
|
// the kernel linker can do runtime patching of vtables.
|
|
|
|
if (Context.getLangOptions().AppleKext)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return !canDevirtualizeMemberFunctionCall(CE->getArg(0), MD);
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Value *
|
|
|
|
CodeGenFunction::EmitCXXOperatorMemberCallee(const CXXOperatorCallExpr *E,
|
|
|
|
const CXXMethodDecl *MD,
|
|
|
|
llvm::Value *This) {
|
|
|
|
const FunctionProtoType *FPT = MD->getType()->castAs<FunctionProtoType>();
|
|
|
|
const llvm::Type *Ty =
|
|
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
|
|
FPT->isVariadic());
|
|
|
|
|
|
|
|
if (UseVirtualCall(getContext(), E, MD))
|
|
|
|
return BuildVirtualCall(MD, This, Ty);
|
|
|
|
|
|
|
|
return CGM.GetAddrOfFunction(MD, Ty);
|
|
|
|
}
|