Sema/AST work for capturing copy init expression

to be used in copy helper synthesis of __block
variables. wip.

llvm-svn: 120617
This commit is contained in:
Fariborz Jahanian 2010-12-01 22:29:46 +00:00
parent 75942f9d93
commit 7cfe7679aa
3 changed files with 44 additions and 0 deletions

View File

@ -129,6 +129,9 @@ class ASTContext {
/// \brief Mapping from ObjCContainers to their ObjCImplementations.
llvm::DenseMap<ObjCContainerDecl*, ObjCImplDecl*> ObjCImpls;
/// \brief Mapping from __block VarDecls to their copy initialization expr.
llvm::DenseMap<VarDecl*, Expr*> BlockVarCopyInits;
/// \brief Representation of a "canonical" template template parameter that
/// is used in canonical template names.
class CanonicalTemplateTemplateParm : public llvm::FoldingSetNode {
@ -1358,6 +1361,12 @@ public:
/// \brief Set the implementation of ObjCCategoryDecl.
void setObjCImplementation(ObjCCategoryDecl *CatD,
ObjCCategoryImplDecl *ImplD);
/// \brief Set the copy inialization expression of a block var decl.
void setBlockVarCopyInits(VarDecl*VD, Expr* Init);
/// \brief Get the copy initialization expression of VarDecl,or NULL if
/// none exists.
Expr *getBlockVarCopyInits(VarDecl*VD);
/// \brief Allocate an uninitialized TypeSourceInfo.
///

View File

@ -1008,6 +1008,20 @@ void ASTContext::setObjCImplementation(ObjCCategoryDecl *CatD,
ObjCImpls[CatD] = ImplD;
}
/// \brief Get the copy initialization expression of VarDecl,or NULL if
/// none exists.
Expr *ASTContext::getBlockVarCopyInits(VarDecl*VD) {
llvm::DenseMap<VarDecl*, Expr*>::iterator
I = BlockVarCopyInits.find(VD);
return (I != BlockVarCopyInits.end()) ? cast<Expr>(I->second) : 0;
}
/// \brief Set the copy inialization expression of a block var decl.
void ASTContext::setBlockVarCopyInits(VarDecl*VD, Expr* Init) {
assert(VD && Init && "Passed null params");
BlockVarCopyInits[VD] = Init;
}
/// \brief Allocate an uninitialized TypeSourceInfo.
///
/// The caller should initialize the memory held by TypeSourceInfo using

View File

@ -3022,6 +3022,27 @@ Sema::ActOnVariableDeclarator(Scope *S, Declarator &D, DeclContext *DC,
if (NewVD->getLinkage() == ExternalLinkage && !DC->isRecord())
AddPushedVisibilityAttribute(NewVD);
// For variables declared as __block which require copy construction,
// must capture copy initialization expression here.
if (getLangOptions().CPlusPlus && NewVD->hasAttr<BlocksAttr>()) {
QualType T = NewVD->getType();
if (!T->isDependentType() && !T->isReferenceType() &&
T->getAs<RecordType>() && !T->isUnionType()) {
Expr *E = new (Context) DeclRefExpr(NewVD, T,
VK_LValue, SourceLocation());
ExprResult Res = PerformCopyInitialization(
InitializedEntity::InitializeBlock(NewVD->getLocation(),
T, false),
SourceLocation(),
Owned(E));
if (!Res.isInvalid()) {
Res = MaybeCreateCXXExprWithTemporaries(Res.get());
Expr *Init = Res.takeAs<Expr>();
Context.setBlockVarCopyInits(NewVD, Init);
}
}
}
MarkUnusedFileScopedDecl(NewVD);
return NewVD;