diff options
author | John McCall <rjmccall@apple.com> | 2011-08-17 21:34:14 +0000 |
---|---|---|
committer | John McCall <rjmccall@apple.com> | 2011-08-17 21:34:14 +0000 |
commit | d963c37eb912342c8325048749e449861cf0a6e3 (patch) | |
tree | 8c3ba183cec56377add9f4102b6c56688aad9fc3 | |
parent | 6704429557544a890cf6c0859b13b1dc63572f70 (diff) |
Gather cleanups correctly in block return statements.
Thanks to Ted for finding this with magic tools.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@137877 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/Sema/SemaStmt.cpp | 19 | ||||
-rw-r--r-- | test/CodeGenCXX/blocks.cpp | 26 |
2 files changed, 33 insertions, 12 deletions
diff --git a/lib/Sema/SemaStmt.cpp b/lib/Sema/SemaStmt.cpp index 7219718ba4..8303616db4 100644 --- a/lib/Sema/SemaStmt.cpp +++ b/lib/Sema/SemaStmt.cpp @@ -1741,21 +1741,17 @@ Sema::ActOnBlockReturnStmt(SourceLocation ReturnLoc, Expr *RetValExp) { // pickier with blocks than for normal functions because we don't have GCC // compatibility to worry about here. ReturnStmt *Result = 0; + const VarDecl *NRVOCandidate = 0; if (CurBlock->ReturnType->isVoidType()) { if (RetValExp && !RetValExp->isTypeDependent() && (!getLangOptions().CPlusPlus || !RetValExp->getType()->isVoidType())) { Diag(ReturnLoc, diag::err_return_block_has_expr); RetValExp = 0; } - Result = new (Context) ReturnStmt(ReturnLoc, RetValExp, 0); } else if (!RetValExp) { if (!CurBlock->ReturnType->isDependentType()) return StmtError(Diag(ReturnLoc, diag::err_block_return_missing_expr)); - - Result = new (Context) ReturnStmt(ReturnLoc, 0, 0); } else { - const VarDecl *NRVOCandidate = 0; - if (!FnRetType->isDependentType() && !RetValExp->isTypeDependent()) { // we have a non-void block with an expression, continue checking @@ -1775,19 +1771,18 @@ Sema::ActOnBlockReturnStmt(SourceLocation ReturnLoc, Expr *RetValExp) { // FIXME: Cleanup temporaries here, anyway? return StmtError(); } + RetValExp = Res.take(); - if (RetValExp) { - CheckImplicitConversions(RetValExp, ReturnLoc); - RetValExp = MaybeCreateExprWithCleanups(RetValExp); - } - - RetValExp = Res.takeAs<Expr>(); if (RetValExp) CheckReturnStackAddr(RetValExp, FnRetType, ReturnLoc); } + } - Result = new (Context) ReturnStmt(ReturnLoc, RetValExp, NRVOCandidate); + if (RetValExp) { + CheckImplicitConversions(RetValExp, ReturnLoc); + RetValExp = MaybeCreateExprWithCleanups(RetValExp); } + Result = new (Context) ReturnStmt(ReturnLoc, RetValExp, NRVOCandidate); // If we need to check for the named return value optimization, save the // return statement in our scope for later processing. diff --git a/test/CodeGenCXX/blocks.cpp b/test/CodeGenCXX/blocks.cpp index 0e310bdbbc..a35b5a370e 100644 --- a/test/CodeGenCXX/blocks.cpp +++ b/test/CodeGenCXX/blocks.cpp @@ -104,3 +104,29 @@ namespace test3 { consume(^{ (void) b; }); } } + +// rdar://problem/9971485 +namespace test4 { + struct A { + A(); + ~A(); + }; + + void foo(A a); + + void test() { + extern void consume(void(^)()); + consume(^{ return foo(A()); }); + } + // CHECK: define void @_ZN5test44testEv() + // CHECK: define internal void @__test_block_invoke + // CHECK: [[TMP:%.*]] = alloca [[A:%.*]], align 1 + // CHECK-NEXT: alloca i32 + // CHECK-NEXT: bitcast i8* + // CHECK-NEXT: call void @_ZN5test41AC1Ev([[A]]* [[TMP]]) + // CHECK-NEXT: call void @_ZN5test43fooENS_1AE([[A]]* [[TMP]]) + // CHECK-NEXT: store i32 1, + // CHECK-NEXT: call void @_ZN5test41AD1Ev([[A]]* [[TMP]]) + // CHECK-NEXT: ret void +} + |