forked from OSchip/llvm-project
For C++ copied in objects, use copy constructors in
setting up block's descriptor. This is on going work to support c++ specific issues in setting up blocks various APIs. llvm-svn: 105469
This commit is contained in:
parent
92eb2cbbef
commit
64176c2c97
|
@ -364,11 +364,28 @@ llvm::Value *CodeGenFunction::BuildBlockLiteralTmp(const BlockExpr *BE) {
|
||||||
E = new (getContext()) DeclRefExpr(const_cast<ValueDecl*>(VD),
|
E = new (getContext()) DeclRefExpr(const_cast<ValueDecl*>(VD),
|
||||||
VD->getType().getNonReferenceType(),
|
VD->getType().getNonReferenceType(),
|
||||||
SourceLocation());
|
SourceLocation());
|
||||||
if (VD->getType()->isReferenceType())
|
if (getContext().getLangOptions().CPlusPlus) {
|
||||||
E = new (getContext())
|
if (VD->getType()->isReferenceType()) {
|
||||||
UnaryOperator(const_cast<Expr*>(E), UnaryOperator::AddrOf,
|
E = new (getContext())
|
||||||
getContext().getPointerType(E->getType()),
|
UnaryOperator(const_cast<Expr*>(E), UnaryOperator::AddrOf,
|
||||||
SourceLocation());
|
getContext().getPointerType(E->getType()),
|
||||||
|
SourceLocation());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
QualType T = E->getType();
|
||||||
|
if (const RecordType *RT = T->getAs<RecordType>()) {
|
||||||
|
CXXRecordDecl *Record = cast<CXXRecordDecl>(RT->getDecl());
|
||||||
|
if (!Record->hasTrivialCopyConstructor()) {
|
||||||
|
CXXConstructorDecl *D = Record->getCopyConstructor(getContext(),
|
||||||
|
0);
|
||||||
|
Expr *Arg = const_cast<Expr*>(E);
|
||||||
|
E = CXXConstructExpr::Create(getContext(), T, D->getLocation(),
|
||||||
|
D, false, &Arg, 1, false,
|
||||||
|
CXXConstructExpr::CK_Complete);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -607,6 +624,10 @@ void CodeGenFunction::AllocateBlockDecl(const BlockDeclRefExpr *E) {
|
||||||
|
|
||||||
llvm::Value *CodeGenFunction::GetAddrOfBlockDecl(const ValueDecl *VD,
|
llvm::Value *CodeGenFunction::GetAddrOfBlockDecl(const ValueDecl *VD,
|
||||||
bool IsByRef) {
|
bool IsByRef) {
|
||||||
|
llvm::Value *&VE = BlockDeclsValue[VD];
|
||||||
|
if (VE)
|
||||||
|
return VE;
|
||||||
|
|
||||||
CharUnits offset = BlockDecls[VD];
|
CharUnits offset = BlockDecls[VD];
|
||||||
assert(!offset.isZero() && "getting address of unallocated decl");
|
assert(!offset.isZero() && "getting address of unallocated decl");
|
||||||
|
|
||||||
|
@ -634,12 +655,12 @@ llvm::Value *CodeGenFunction::GetAddrOfBlockDecl(const ValueDecl *VD,
|
||||||
V = Builder.CreateLoad(V);
|
V = Builder.CreateLoad(V);
|
||||||
} else {
|
} else {
|
||||||
const llvm::Type *Ty = CGM.getTypes().ConvertType(VD->getType());
|
const llvm::Type *Ty = CGM.getTypes().ConvertType(VD->getType());
|
||||||
|
|
||||||
Ty = llvm::PointerType::get(Ty, 0);
|
Ty = llvm::PointerType::get(Ty, 0);
|
||||||
V = Builder.CreateBitCast(V, Ty);
|
V = Builder.CreateBitCast(V, Ty);
|
||||||
if (VD->getType()->isReferenceType())
|
if (VD->getType()->isReferenceType())
|
||||||
V = Builder.CreateLoad(V, "tmp");
|
V = Builder.CreateLoad(V, "ref.tmp");
|
||||||
}
|
}
|
||||||
|
VE = V;
|
||||||
return V;
|
return V;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -180,6 +180,9 @@ public:
|
||||||
|
|
||||||
/// BlockDecls - Offsets for all Decls in BlockDeclRefExprs.
|
/// BlockDecls - Offsets for all Decls in BlockDeclRefExprs.
|
||||||
llvm::DenseMap<const Decl*, CharUnits> BlockDecls;
|
llvm::DenseMap<const Decl*, CharUnits> BlockDecls;
|
||||||
|
|
||||||
|
/// BlockDeclsValue - llvm::Value for all Decls in BlockDeclRefExprs.
|
||||||
|
llvm::DenseMap<const Decl*, llvm::Value *> BlockDeclsValue;
|
||||||
|
|
||||||
/// BlockCXXThisOffset - The offset of the C++ 'this' value within
|
/// BlockCXXThisOffset - The offset of the C++ 'this' value within
|
||||||
/// the block structure.
|
/// the block structure.
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
// RUN: %clang_cc1 %s -fblocks -triple x86_64-apple-darwin -emit-llvm -o - | FileCheck %s
|
||||||
|
|
||||||
|
struct TestObject
|
||||||
|
{
|
||||||
|
TestObject(const TestObject& inObj);
|
||||||
|
TestObject();
|
||||||
|
TestObject& operator=(const TestObject& inObj);
|
||||||
|
int version() const;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
void testRoutine() {
|
||||||
|
TestObject one;
|
||||||
|
int (^V)() = ^{ return one.version(); };
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: call void @_ZN10TestObjectC1ERKS_
|
||||||
|
|
Loading…
Reference in New Issue