; RUN: opt -S -objc-arc < %s | FileCheck %s ; rdar://10209613 %0 = type opaque %struct.__block_descriptor = type { i64, i64 } @_NSConcreteStackBlock = external global i8* @__block_descriptor_tmp = external hidden constant { i64, i64, i8*, i8*, i8*, i8* } @"\01L_OBJC_SELECTOR_REFERENCES_" = external hidden global i8*, section "__DATA, __objc_selrefs, literal_pointers, no_dead_strip" ; CHECK: define void @test( ; CHECK: %3 = call i8* @objc_retainBlock(i8* %2) [[NUW:#[0-9]+]] ; CHECK: @objc_msgSend ; CHECK-NEXT: @objc_release(i8* %3) define void @test(%0* %array) uwtable { entry: %block = alloca <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>, align 8 %0 = bitcast %0* %array to i8* %1 = tail call i8* @objc_retain(i8* %0) nounwind %block.isa = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 0 store i8* bitcast (i8** @_NSConcreteStackBlock to i8*), i8** %block.isa, align 8 %block.flags = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 1 store i32 1107296256, i32* %block.flags, align 8 %block.reserved = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 2 store i32 0, i32* %block.reserved, align 4 %block.invoke = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 3 store i8* bitcast (void (i8*)* @__test_block_invoke_0 to i8*), i8** %block.invoke, align 8 %block.descriptor = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 4 store %struct.__block_descriptor* bitcast ({ i64, i64, i8*, i8*, i8*, i8* }* @__block_descriptor_tmp to %struct.__block_descriptor*), %struct.__block_descriptor** %block.descriptor, align 8 %block.captured = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 5 store %0* %array, %0** %block.captured, align 8 %2 = bitcast <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block to i8* %3 = call i8* @objc_retainBlock(i8* %2) nounwind %tmp2 = load i8** @"\01L_OBJC_SELECTOR_REFERENCES_", align 8 call void bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to void (i8*, i8*, i8*)*)(i8* %0, i8* %tmp2, i8* %3) call void @objc_release(i8* %3) nounwind %strongdestroy = load %0** %block.captured, align 8 %4 = bitcast %0* %strongdestroy to i8* call void @objc_release(i8* %4) nounwind, !clang.imprecise_release !0 ret void } ; Same as test, but the objc_retainBlock has a clang.arc.copy_on_escape ; tag so it's safe to delete. ; CHECK: define void @test_with_COE( ; CHECK-NOT: @objc_retainBlock ; CHECK: @objc_msgSend ; CHECK: @objc_release ; CHECK-NOT: @objc_release ; CHECK: } define void @test_with_COE(%0* %array) uwtable { entry: %block = alloca <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>, align 8 %0 = bitcast %0* %array to i8* %1 = tail call i8* @objc_retain(i8* %0) nounwind %block.isa = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 0 store i8* bitcast (i8** @_NSConcreteStackBlock to i8*), i8** %block.isa, align 8 %block.flags = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 1 store i32 1107296256, i32* %block.flags, align 8 %block.reserved = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 2 store i32 0, i32* %block.reserved, align 4 %block.invoke = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 3 store i8* bitcast (void (i8*)* @__test_block_invoke_0 to i8*), i8** %block.invoke, align 8 %block.descriptor = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 4 store %struct.__block_descriptor* bitcast ({ i64, i64, i8*, i8*, i8*, i8* }* @__block_descriptor_tmp to %struct.__block_descriptor*), %struct.__block_descriptor** %block.descriptor, align 8 %block.captured = getelementptr inbounds <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block, i64 0, i32 5 store %0* %array, %0** %block.captured, align 8 %2 = bitcast <{ i8*, i32, i32, i8*, %struct.__block_descriptor*, %0* }>* %block to i8* %3 = call i8* @objc_retainBlock(i8* %2) nounwind, !clang.arc.copy_on_escape !0 %tmp2 = load i8** @"\01L_OBJC_SELECTOR_REFERENCES_", align 8 call void bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to void (i8*, i8*, i8*)*)(i8* %0, i8* %tmp2, i8* %3) call void @objc_release(i8* %3) nounwind %strongdestroy = load %0** %block.captured, align 8 %4 = bitcast %0* %strongdestroy to i8* call void @objc_release(i8* %4) nounwind, !clang.imprecise_release !0 ret void } declare i8* @objc_retain(i8*) declare void @__test_block_invoke_0(i8* nocapture) uwtable declare i8* @objc_retainBlock(i8*) declare i8* @objc_msgSend(i8*, i8*, ...) nonlazybind declare void @objc_release(i8*) ; CHECK: attributes #0 = { uwtable } ; CHECK: attributes #1 = { nonlazybind } ; CHECK: attributes [[NUW]] = { nounwind } !0 = metadata !{}