diff options
| author | John McCall <rjmccall@apple.com> | 2013-03-04 06:32:36 +0000 |
|---|---|---|
| committer | John McCall <rjmccall@apple.com> | 2013-03-04 06:32:36 +0000 |
| commit | a37c2fa40966b75624a524913bddfc4ea669bb46 (patch) | |
| tree | b4e66fa2dbfdca72ff3a345d4e062b53a669d692 /clang | |
| parent | 7b0ae11c8265f6d42bbdd02eacd1c8c07cb2fcc5 (diff) | |
| download | bcm5719-llvm-a37c2fa40966b75624a524913bddfc4ea669bb46.tar.gz bcm5719-llvm-a37c2fa40966b75624a524913bddfc4ea669bb46.zip | |
Fix the emission of the copy-initialization of a block capture
from a lambda capture when the capture is not trivially-copyable.
rdar://13295759
llvm-svn: 176431
Diffstat (limited to 'clang')
| -rw-r--r-- | clang/lib/CodeGen/CGBlocks.cpp | 12 | ||||
| -rw-r--r-- | clang/test/CodeGenCXX/blocks-cxx11.cpp | 30 |
2 files changed, 40 insertions, 2 deletions
diff --git a/clang/lib/CodeGen/CGBlocks.cpp b/clang/lib/CodeGen/CGBlocks.cpp index 23aa066dcea..b9f466117c6 100644 --- a/clang/lib/CodeGen/CGBlocks.cpp +++ b/clang/lib/CodeGen/CGBlocks.cpp @@ -777,8 +777,16 @@ llvm::Value *CodeGenFunction::EmitBlockLiteral(const CGBlockInfo &blockInfo) { // special; we'll simply emit it directly. src = 0; } else { - // This is a [[type]]*. - src = LocalDeclMap[variable]; + // Just look it up in the locals map, which will give us back a + // [[type]]*. If that doesn't work, do the more elaborate DRE + // emission. + src = LocalDeclMap.lookup(variable); + if (!src) { + DeclRefExpr declRef(const_cast<VarDecl*>(variable), + /*refersToEnclosing*/ ci->isNested(), type, + VK_LValue, SourceLocation()); + src = EmitDeclRefLValue(&declRef).getAddress(); + } } // For byrefs, we just write the pointer to the byref struct into diff --git a/clang/test/CodeGenCXX/blocks-cxx11.cpp b/clang/test/CodeGenCXX/blocks-cxx11.cpp index 996db1afe69..3f0380abbd3 100644 --- a/clang/test/CodeGenCXX/blocks-cxx11.cpp +++ b/clang/test/CodeGenCXX/blocks-cxx11.cpp @@ -82,3 +82,33 @@ namespace test_complex_int_ref_mutable { } } +// rdar://13295759 +namespace test_block_in_lambda { + void takeBlock(void (^block)()); + + // The captured variable has to be non-POD so that we have a copy expression. + struct A { + void *p; + A(const A &); + ~A(); + void use() const; + }; + + void test(A a) { + auto lambda = [a]() { + takeBlock(^{ a.use(); }); + }; + lambda(); // make sure we emit the invocation function + } + // CHECK: define internal void @"_ZZN20test_block_in_lambda4testENS_1AEENK3$_0clEv"( + // CHECK: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], align 8 + // CHECK: [[THIS:%.*]] = load [[LAMBDA_T:%.*]]** + // CHECK: [[TO_DESTROY:%.*]] = getelementptr inbounds [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 + // CHECK: [[T0:%.*]] = getelementptr inbounds [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 + // CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds [[LAMBDA_T]]* [[THIS]], i32 0, i32 0 + // CHECK-NEXT: call void @_ZN20test_block_in_lambda1AC1ERKS0_({{.*}}* [[T0]], {{.*}}* [[T1]]) + // CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to void ()* + // CHECK-NEXT: call void @_ZN20test_block_in_lambda9takeBlockEU13block_pointerFvvE(void ()* [[T0]]) + // CHECK-NEXT: call void @_ZN20test_block_in_lambda1AD1Ev({{.*}}* [[TO_DESTROY]]) + // CHECK-NEXT: ret void +} |

