From ec79d877c1998366480d97a7a6c94e15c053edd8 Mon Sep 17 00:00:00 2001 From: Douglas Gregor Date: Fri, 24 Feb 2012 17:41:38 +0000 Subject: [PATCH] Two minor, related fixes for template instantiation with blocks: - Make sure that the block expression is instantiation-dependent if the block is in a dependent context - Make sure that the C++ 'this' expression gets captured even if we don't rebuild the AST node during template instantiation. This would also have manifested as a bug for lambdas. Fixes . git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@151372 91177308-0d34-0410-b5e6-96231b3b80d8 --- include/clang/AST/Expr.h | 5 ++--- lib/Sema/TreeTransform.h | 8 ++++++-- test/CodeGenCXX/blocks.cpp | 15 +++++++++++++++ 3 files changed, 23 insertions(+), 5 deletions(-) diff --git a/include/clang/AST/Expr.h b/include/clang/AST/Expr.h index b9c3220c0b..741b47e560 100644 --- a/include/clang/AST/Expr.h +++ b/include/clang/AST/Expr.h @@ -4220,9 +4220,8 @@ protected: public: BlockExpr(BlockDecl *BD, QualType ty) : Expr(BlockExprClass, ty, VK_RValue, OK_Ordinary, - ty->isDependentType(), false, - // FIXME: Check for instantiate-dependence in the statement? - ty->isInstantiationDependentType(), + ty->isDependentType(), ty->isDependentType(), + ty->isInstantiationDependentType() || BD->isDependentContext(), false), TheBlock(BD) {} diff --git a/lib/Sema/TreeTransform.h b/lib/Sema/TreeTransform.h index f0b5f4e241..11f03fc133 100644 --- a/lib/Sema/TreeTransform.h +++ b/lib/Sema/TreeTransform.h @@ -7038,9 +7038,12 @@ TreeTransform::TransformCXXThisExpr(CXXThisExpr *E) { T = getSema().Context.getPointerType( getSema().Context.getRecordType(cast(DC))); - if (!getDerived().AlwaysRebuild() && T == E->getType()) + if (!getDerived().AlwaysRebuild() && T == E->getType()) { + // Make sure that we capture 'this'. + getSema().CheckCXXThisCapture(E->getLocStart()); return SemaRef.Owned(E); - + } + return getDerived().RebuildCXXThisExpr(E->getLocStart(), T, E->isImplicit()); } @@ -8539,6 +8542,7 @@ TreeTransform::TransformBlockExpr(BlockExpr *E) { oldCapture)); assert(blockScope->CaptureMap.count(newCapture)); } + assert(oldBlock->capturesCXXThis() == blockScope->isCXXThisCaptured()); } #endif diff --git a/test/CodeGenCXX/blocks.cpp b/test/CodeGenCXX/blocks.cpp index 3d7fcfddbe..eb544787da 100644 --- a/test/CodeGenCXX/blocks.cpp +++ b/test/CodeGenCXX/blocks.cpp @@ -211,3 +211,18 @@ namespace test7 { return ^{ return *p; }(); } } + +namespace test8 { + // : failure to capture this after skipping rebuild + // of the 'this' pointer. + struct X { + int x; + + template + int foo() { + return ^ { return x; }(); + } + }; + + template int X::foo(); +} -- 2.40.0