diff options
author | Douglas Gregor <dgregor@apple.com> | 2011-06-22 16:12:01 +0000 |
---|---|---|
committer | Douglas Gregor <dgregor@apple.com> | 2011-06-22 16:12:01 +0000 |
commit | 58df509fc0dc288f6859cd3b61ff980d364baf6d (patch) | |
tree | 83ec8750fc9ef7b24b397a85eab8f3bc37bfbcab /clang/lib/CodeGen/CGObjC.cpp | |
parent | 08d0f3550a87f04fdd75e118eed163c441e2e494 (diff) | |
download | bcm5719-llvm-58df509fc0dc288f6859cd3b61ff980d364baf6d.tar.gz bcm5719-llvm-58df509fc0dc288f6859cd3b61ff980d364baf6d.zip |
When binding a reference to an Automatic Reference Counting temporary,
retain/release the temporary object appropriately. Previously, we
would only perform the retain/release operations when the reference
would extend the lifetime of the temporary, but this does the wrong
thing across calls.
llvm-svn: 133620
Diffstat (limited to 'clang/lib/CodeGen/CGObjC.cpp')
-rw-r--r-- | clang/lib/CodeGen/CGObjC.cpp | 46 |
1 files changed, 42 insertions, 4 deletions
diff --git a/clang/lib/CodeGen/CGObjC.cpp b/clang/lib/CodeGen/CGObjC.cpp index ede2302403c..a43f4511096 100644 --- a/clang/lib/CodeGen/CGObjC.cpp +++ b/clang/lib/CodeGen/CGObjC.cpp @@ -1985,6 +1985,21 @@ namespace { CallReleaseForObject(QualType type, llvm::Value *addr, bool precise) : ObjCReleasingCleanup(type, addr), precise(precise) {} + using ObjCReleasingCleanup::Emit; + static void Emit(CodeGenFunction &CGF, bool IsForEH, + QualType type, llvm::Value *addr, bool precise) { + // EHScopeStack::Cleanup objects can never have their destructors called, + // so use placement new to construct our temporary object. + union { + void* align; + char data[sizeof(CallReleaseForObject)]; + }; + + CallReleaseForObject *Object + = new (&align) CallReleaseForObject(type, addr, precise); + Object->Emit(CGF, IsForEH); + } + void release(CodeGenFunction &CGF, QualType type, llvm::Value *addr) { llvm::Value *ptr = CGF.Builder.CreateLoad(addr, "tmp"); CGF.EmitARCRelease(ptr, precise); @@ -2033,6 +2048,21 @@ namespace { CallWeakReleaseForObject(QualType type, llvm::Value *addr) : ObjCReleasingCleanup(type, addr) {} + using ObjCReleasingCleanup::Emit; + static void Emit(CodeGenFunction &CGF, bool IsForEH, + QualType type, llvm::Value *addr) { + // EHScopeStack::Cleanup objects can never have their destructors called, + // so use placement new to construct our temporary object. + union { + void* align; + char data[sizeof(CallWeakReleaseForObject)]; + }; + + CallWeakReleaseForObject *Object + = new (&align) CallWeakReleaseForObject(type, addr); + Object->Emit(CGF, IsForEH); + } + void release(CodeGenFunction &CGF, QualType type, llvm::Value *addr) { CGF.EmitARCDestroyWeak(addr); } @@ -2099,16 +2129,24 @@ void CodeGenFunction::EmitObjCAutoreleasePoolCleanup(llvm::Value *Ptr) { void CodeGenFunction::PushARCReleaseCleanup(CleanupKind cleanupKind, QualType type, llvm::Value *addr, - bool precise) { - EHStack.pushCleanup<CallReleaseForObject>(cleanupKind, type, addr, precise); + bool precise, + bool forFullExpr) { + if (forFullExpr) + pushFullExprCleanup<CallReleaseForObject>(cleanupKind, type, addr, precise); + else + EHStack.pushCleanup<CallReleaseForObject>(cleanupKind, type, addr, precise); } /// PushARCWeakReleaseCleanup - Enter a cleanup to perform a weak /// release on the given object or array of objects. void CodeGenFunction::PushARCWeakReleaseCleanup(CleanupKind cleanupKind, QualType type, - llvm::Value *addr) { - EHStack.pushCleanup<CallWeakReleaseForObject>(cleanupKind, type, addr); + llvm::Value *addr, + bool forFullExpr) { + if (forFullExpr) + pushFullExprCleanup<CallWeakReleaseForObject>(cleanupKind, type, addr); + else + EHStack.pushCleanup<CallWeakReleaseForObject>(cleanupKind, type, addr); } /// PushARCReleaseCleanup - Enter a cleanup to perform a release on a |