; RUN: opt -objc-arc -S < %s | FileCheck %s target datalayout = "e-p:64:64:64" !0 = metadata !{} declare i8* @objc_retain(i8*) declare void @callee(i8) declare void @use_pointer(i8*) declare void @objc_release(i8*) declare i8* @objc_retainBlock(i8*) declare i8* @objc_autorelease(i8*) ; Basic retainBlock+release elimination. ; CHECK-LABEL: define void @test0(i8* %tmp) { ; CHECK-NOT: @objc ; CHECK: } define void @test0(i8* %tmp) { entry: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind, !clang.arc.copy_on_escape !0 tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 ret void } ; Same as test0, but there's no copy_on_escape metadata, so there's no ; optimization possible. ; CHECK-LABEL: define void @test0_no_metadata(i8* %tmp) { ; CHECK: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) [[NUW:#[0-9]+]] ; CHECK: tail call void @objc_release(i8* %tmp2) [[NUW]], !clang.imprecise_release !0 ; CHECK: } define void @test0_no_metadata(i8* %tmp) { entry: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 ret void } ; Same as test0, but the pointer escapes, so there's no ; optimization possible. ; CHECK-LABEL: define void @test0_escape(i8* %tmp, i8** %z) { ; CHECK: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) [[NUW]], !clang.arc.copy_on_escape !0 ; CHECK: tail call void @objc_release(i8* %tmp2) [[NUW]], !clang.imprecise_release !0 ; CHECK: } define void @test0_escape(i8* %tmp, i8** %z) { entry: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind, !clang.arc.copy_on_escape !0 store i8* %tmp2, i8** %z tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 ret void } ; Same as test0_escape, but there's no intervening call. ; CHECK-LABEL: define void @test0_just_escape(i8* %tmp, i8** %z) { ; CHECK: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) [[NUW]], !clang.arc.copy_on_escape !0 ; CHECK: tail call void @objc_release(i8* %tmp2) [[NUW]], !clang.imprecise_release !0 ; CHECK: } define void @test0_just_escape(i8* %tmp, i8** %z) { entry: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind, !clang.arc.copy_on_escape !0 store i8* %tmp2, i8** %z tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 ret void } ; Basic nested retainBlock+release elimination. ; CHECK: define void @test1(i8* %tmp) { ; CHECK-NOT: @objc ; CHECK: tail call i8* @objc_retain(i8* %tmp) [[NUW]] ; CHECK-NOT: @objc ; CHECK: tail call void @objc_release(i8* %tmp) [[NUW]], !clang.imprecise_release !0 ; CHECK-NOT: @objc ; CHECK: } define void @test1(i8* %tmp) { entry: %tmp1 = tail call i8* @objc_retain(i8* %tmp) nounwind %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind, !clang.arc.copy_on_escape !0 tail call void @use_pointer(i8* %tmp2) tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 tail call void @objc_release(i8* %tmp) nounwind, !clang.imprecise_release !0 ret void } ; Same as test1, but there's no copy_on_escape metadata, so there's no ; retainBlock+release optimization possible. But we can still eliminate ; the outer retain+release. ; CHECK-LABEL: define void @test1_no_metadata(i8* %tmp) { ; CHECK-NEXT: entry: ; CHECK-NEXT: tail call i8* @objc_retainBlock(i8* %tmp) [[NUW]] ; CHECK-NEXT: @use_pointer(i8* %tmp2) ; CHECK-NEXT: @use_pointer(i8* %tmp2) ; CHECK-NEXT: tail call void @objc_release(i8* %tmp2) [[NUW]], !clang.imprecise_release !0 ; CHECK-NOT: @objc ; CHECK: } define void @test1_no_metadata(i8* %tmp) { entry: %tmp1 = tail call i8* @objc_retain(i8* %tmp) nounwind %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind tail call void @use_pointer(i8* %tmp2) tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 tail call void @objc_release(i8* %tmp) nounwind, !clang.imprecise_release !0 ret void } ; Same as test1, but the pointer escapes, so there's no ; retainBlock+release optimization possible. But we can still eliminate ; the outer retain+release ; CHECK: define void @test1_escape(i8* %tmp, i8** %z) { ; CHECK-NEXT: entry: ; CHECK-NEXT: %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) [[NUW]], !clang.arc.copy_on_escape !0 ; CHECK-NEXT: store i8* %tmp2, i8** %z ; CHECK-NEXT: @use_pointer(i8* %tmp2) ; CHECK-NEXT: @use_pointer(i8* %tmp2) ; CHECK-NEXT: tail call void @objc_release(i8* %tmp2) [[NUW]], !clang.imprecise_release !0 ; CHECK-NOT: @objc ; CHECK: } define void @test1_escape(i8* %tmp, i8** %z) { entry: %tmp1 = tail call i8* @objc_retain(i8* %tmp) nounwind %tmp2 = tail call i8* @objc_retainBlock(i8* %tmp) nounwind, !clang.arc.copy_on_escape !0 store i8* %tmp2, i8** %z tail call void @use_pointer(i8* %tmp2) tail call void @use_pointer(i8* %tmp2) tail call void @objc_release(i8* %tmp2) nounwind, !clang.imprecise_release !0 tail call void @objc_release(i8* %tmp) nounwind, !clang.imprecise_release !0 ret void } ; Optimize objc_retainBlock. ; CHECK-LABEL: define void @test23( ; CHECK-NOT: @objc_ ; CHECK: } %block0 = type { i64, i64, i8*, i8* } %block1 = type { i8**, i32, i32, i32 (%struct.__block_literal_1*)*, %block0* } %struct.__block_descriptor = type { i64, i64 } %struct.__block_literal_1 = type { i8**, i32, i32, i8**, %struct.__block_descriptor* } @__block_holder_tmp_1 = external constant %block1 define void @test23() { entry: %0 = call i8* @objc_retainBlock(i8* bitcast (%block1* @__block_holder_tmp_1 to i8*)) nounwind, !clang.arc.copy_on_escape !0 call void @bar(i32 ()* bitcast (%block1* @__block_holder_tmp_1 to i32 ()*)) call void @bar(i32 ()* bitcast (%block1* @__block_holder_tmp_1 to i32 ()*)) call void @objc_release(i8* bitcast (%block1* @__block_holder_tmp_1 to i8*)) nounwind ret void } ; Don't optimize objc_retainBlock, but do strength reduce it. ; CHECK-LABEL: define void @test3a(i8* %p) { ; CHECK: @objc_retain ; CHECK: @objc_release ; CHECK: } define void @test3a(i8* %p) { entry: %0 = call i8* @objc_retainBlock(i8* %p) nounwind, !clang.arc.copy_on_escape !0 call void @callee() call void @use_pointer(i8* %p) call void @objc_release(i8* %p) nounwind ret void } ; Don't optimize objc_retainBlock, because there's no copy_on_escape metadata. ; CHECK-LABEL: define void @test3b( ; CHECK: @objc_retainBlock ; CHECK: @objc_release ; CHECK: } define void @test3b() { entry: %0 = call i8* @objc_retainBlock(i8* bitcast (%block1* @__block_holder_tmp_1 to i8*)) nounwind call void @bar(i32 ()* bitcast (%block1* @__block_holder_tmp_1 to i32 ()*)) call void @bar(i32 ()* bitcast (%block1* @__block_holder_tmp_1 to i32 ()*)) call void @objc_release(i8* bitcast (%block1* @__block_holder_tmp_1 to i8*)) nounwind ret void } ; CHECK: attributes [[NUW]] = { nounwind }