; RUN: %swift-llvm-opt -swift-llvm-arc-optimize %s | %FileCheck %s target datalayout = "e-p:64:64:64-S128-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f16:16:16-f32:32:32-f64:64:64-f128:128:128-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64" target triple = "x86_64-apple-macosx10.9" %swift.refcounted = type { %swift.heapmetadata*, i64 } %swift.heapmetadata = type { i64 (%swift.refcounted*)*, i64 (%swift.refcounted*)* } %objc_object = type opaque %swift.bridge = type opaque declare void @swift_unknownRetain(%swift.refcounted*) declare void @swift_unknownRelease(%swift.refcounted*) declare %objc_object* @objc_retain(%objc_object*) declare void @objc_release(%objc_object*) declare %swift.refcounted* @swift_allocObject(%swift.heapmetadata* , i64, i64) nounwind declare void @swift_release(%swift.refcounted* nocapture) declare void @swift_retain(%swift.refcounted* ) nounwind declare %swift.bridge* @swift_bridgeObjectRetain(%swift.bridge*) declare void @swift_bridgeObjectRelease(%swift.bridge*) declare void @swift_retainUnowned(%swift.refcounted*) declare void @user(%swift.refcounted *) nounwind declare void @user_objc(%objc_object*) nounwind declare void @unknown_func() define private void @__swift_fixLifetime(%swift.refcounted*) noinline nounwind { entry: ret void } ; CHECK-LABEL: @trivial_objc_canonicalization( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[RET0:%.+]] = bitcast i8* %O to %objc_object* ; CHECK-NEXT: [[RET1:%.+]] = tail call %objc_object* @objc_retain(%objc_object* [[RET0:%.+]]) ; CHECK-NEXT: call void @user_objc(%objc_object* [[RET0:%.+]]) ; CHECK-NEXT: ret void define void @trivial_objc_canonicalization(i8* %O) { entry: %0 = bitcast i8* %O to %objc_object* %1 = tail call %objc_object* @objc_retain(%objc_object* %0) call void @user_objc(%objc_object* %1) nounwind ret void } ; CHECK-LABEL: @trivial_retain_release( ; CHECK-NEXT: entry: ; CHECK-NEXT: call void @user ; CHECK-NEXT: ret void define void @trivial_retain_release(%swift.refcounted* %P, %objc_object* %O, %swift.bridge * %B) { entry: tail call void @swift_retain(%swift.refcounted* %P) tail call void @swift_release(%swift.refcounted* %P) nounwind tail call void @swift_unknownRetain(%swift.refcounted* %P) tail call void @swift_unknownRelease(%swift.refcounted* %P) tail call %objc_object* @objc_retain(%objc_object* %O) tail call void @objc_release(%objc_object* %O) %v = tail call %swift.bridge* @swift_bridgeObjectRetain(%swift.bridge* %B) tail call void @swift_bridgeObjectRelease(%swift.bridge* %v) call void @user(%swift.refcounted* %P) nounwind ret void } ; CHECK-LABEL: @trivial_retain_release_with_rcidentity( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[RET0:%.+]] = bitcast %swift.refcounted* %P to %swift.refcounted* ; CHECK-NEXT: [[RET1:%.+]] = bitcast %swift.refcounted* %P to %swift.refcounted* ; CHECK-NEXT: [[RET2:%.+]] = bitcast %objc_object* %O to %objc_object* ; CHECK-NEXT: call void @user ; CHECK-NEXT: ret void define void @trivial_retain_release_with_rcidentity(%swift.refcounted* %P, %objc_object* %O, %swift.bridge * %B) { entry: tail call void @swift_retain(%swift.refcounted* %P) %0 = bitcast %swift.refcounted* %P to %swift.refcounted* tail call void @swift_release(%swift.refcounted* %0) nounwind tail call void @swift_unknownRetain(%swift.refcounted* %P) %1 = bitcast %swift.refcounted* %P to %swift.refcounted* tail call void @swift_unknownRelease(%swift.refcounted* %1) tail call %objc_object* @objc_retain(%objc_object* %O) %3 = bitcast %objc_object* %O to %objc_object* tail call void @objc_release(%objc_object* %3) call void @user(%swift.refcounted* %P) nounwind ret void } ; retain_motion1 - This shows motion of a retain across operations that can't ; release an object. Release motion can't zap this. ; CHECK-LABEL: @retain_motion1( ; CHECK-NEXT: bitcast ; CHECK-NEXT: store i32 ; CHECK-NEXT: ret void define void @retain_motion1(%swift.refcounted* %A) { tail call void @swift_retain(%swift.refcounted* %A) %B = bitcast %swift.refcounted* %A to i32* store i32 42, i32* %B tail call void @swift_release(%swift.refcounted* %A) nounwind ret void } ; rdar://11583269 - Optimize out objc_retain/release(null) ; CHECK-LABEL: @objc_retain_release_null( ; CHECK-NEXT: entry: ; CHECK-NEXT: ret void define void @objc_retain_release_null() { entry: tail call void @objc_release(%objc_object* null) nounwind tail call %objc_object* @objc_retain(%objc_object* null) ret void } ; CHECK-LABEL: @swiftunknown_retain_release_null( ; CHECK-NEXT: entry: ; CHECK-NEXT: ret void define void @swiftunknown_retain_release_null() { entry: tail call void @swift_unknownRelease(%swift.refcounted* null) tail call void @swift_unknownRetain(%swift.refcounted* null) nounwind ret void } ; rdar://11583269 - Useless objc_retain/release optimization. ; CHECK-LABEL: @objc_retain_release_opt( ; CHECK-NEXT: store i32 42 ; CHECK-NEXT: ret void define void @objc_retain_release_opt(%objc_object* %P, i32* %IP) { tail call %objc_object* @objc_retain(%objc_object* %P) nounwind store i32 42, i32* %IP tail call void @objc_release(%objc_object* %P) nounwind ret void } ; CHECK-LABEL: define{{( protected)?}} void @swift_fixLifetimeTest ; CHECK: swift_retain ; CHECK: swift_fixLifetime ; CHECK: swift_release define void @swift_fixLifetimeTest(%swift.refcounted* %A) { tail call void @swift_retain(%swift.refcounted* %A) call void @user(%swift.refcounted* %A) nounwind call void @__swift_fixLifetime(%swift.refcounted* %A) tail call void @swift_release(%swift.refcounted* %A) nounwind ret void } ; CHECK-LABEL: @move_retain_across_unknown_retain ; CHECK-NOT: swift_retain ; CHECK: swift_unknownRetain ; CHECK-NOT: swift_release ; CHECK: ret define void @move_retain_across_unknown_retain(%swift.refcounted* %A, %swift.refcounted* %B) { tail call void @swift_retain(%swift.refcounted* %A) tail call void @swift_unknownRetain(%swift.refcounted* %B) tail call void @swift_release(%swift.refcounted* %A) nounwind ret void } ; CHECK-LABEL: @move_retain_across_objc_retain ; CHECK-NOT: swift_retain ; CHECK: objc_retain ; CHECK-NOT: swift_release ; CHECK: ret define void @move_retain_across_objc_retain(%swift.refcounted* %A, %objc_object* %B) { tail call void @swift_retain(%swift.refcounted* %A) tail call %objc_object* @objc_retain(%objc_object* %B) tail call void @swift_release(%swift.refcounted* %A) nounwind ret void } ; CHECK-LABEL: @move_retain_across_load ; CHECK-NOT: swift_retain ; CHECK-NOT: swift_release ; CHECK: ret define i32 @move_retain_across_load(%swift.refcounted* %A, i32* %ptr) { tail call void @swift_retain(%swift.refcounted* %A) %val = load i32, i32* %ptr tail call void @swift_release(%swift.refcounted* %A) nounwind ret i32 %val } ; CHECK-LABEL: @move_retain_but_not_release_across_objc_fix_lifetime ; CHECK: call void @__swift_fixLifetime ; CHECK-NEXT: tail call void @swift_retain ; CHECK-NEXT: call void @user ; CHECK-NEXT: call void @__swift_fixLifetime ; CHECK-NEXT: call void @swift_release ; CHECK-NEXT: ret define void @move_retain_but_not_release_across_objc_fix_lifetime(%swift.refcounted* %A) { tail call void @swift_retain(%swift.refcounted* %A) call void @__swift_fixLifetime(%swift.refcounted* %A) nounwind call void @user(%swift.refcounted* %A) nounwind call void @__swift_fixLifetime(%swift.refcounted* %A) nounwind tail call void @swift_release(%swift.refcounted* %A) nounwind ret void } ; CHECK-LABEL: @optimize_retain_unowned ; CHECK-NEXT: bitcast ; CHECK-NEXT: load ; CHECK-NEXT: add ; CHECK-NEXT: load ; CHECK-NEXT: call void @swift_checkUnowned ; CHECK-NEXT: ret define void @optimize_retain_unowned(%swift.refcounted* %A) { tail call void @swift_retainUnowned(%swift.refcounted* %A) %value = bitcast %swift.refcounted* %A to i64* ; loads from the %A and speculatively executable instructions %L1 = load i64, i64* %value, align 8 %R1 = add i64 %L1, 1 %L2 = load i64, i64* %value, align 8 tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @dont_optimize_retain_unowned ; CHECK-NEXT: call void @swift_retainUnowned ; CHECK-NEXT: bitcast ; CHECK-NEXT: load ; CHECK-NEXT: load ; CHECK-NEXT: call void @swift_release ; CHECK-NEXT: ret define void @dont_optimize_retain_unowned(%swift.refcounted* %A) { tail call void @swift_retainUnowned(%swift.refcounted* %A) %value = bitcast %swift.refcounted* %A to i64** %L1 = load i64*, i64** %value, align 8 ; Use of a potential garbage address from a load of %A. %L2 = load i64, i64* %L1, align 8 tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @dont_optimize_retain_unowned2 ; CHECK-NEXT: call void @swift_retainUnowned ; CHECK-NEXT: store ; CHECK-NEXT: call void @swift_release ; CHECK-NEXT: ret define void @dont_optimize_retain_unowned2(%swift.refcounted* %A, i32* %B) { tail call void @swift_retainUnowned(%swift.refcounted* %A) ; store to an unknown address store i32 42, i32* %B tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @dont_optimize_retain_unowned3 ; CHECK-NEXT: call void @swift_retainUnowned ; CHECK-NEXT: call void @unknown_func ; CHECK-NEXT: call void @swift_release ; CHECK-NEXT: ret define void @dont_optimize_retain_unowned3(%swift.refcounted* %A) { tail call void @swift_retainUnowned(%swift.refcounted* %A) ; call of an unknown function call void @unknown_func() tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @dont_optimize_retain_unowned4 ; CHECK-NEXT: call void @swift_retainUnowned ; CHECK-NEXT: call void @swift_retain ; CHECK-NEXT: call void @swift_release ; CHECK-NEXT: ret define void @dont_optimize_retain_unowned4(%swift.refcounted* %A, %swift.refcounted* %B) { tail call void @swift_retainUnowned(%swift.refcounted* %A) ; retain of an unknown reference (%B could be equal to %A) tail call void @swift_retain(%swift.refcounted* %B) tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @remove_redundant_check_unowned ; CHECK-NEXT: bitcast ; CHECK-NEXT: load ; CHECK-NEXT: call void @swift_checkUnowned ; CHECK-NEXT: load ; CHECK-NEXT: store ; CHECK-NEXT: load ; CHECK-NEXT: ret define void @remove_redundant_check_unowned(%swift.refcounted* %A, %swift.refcounted* %B, i64* %C) { tail call void @swift_retainUnowned(%swift.refcounted* %A) %addr = bitcast %swift.refcounted* %A to i64* %L1 = load i64, i64* %addr, align 8 tail call void @swift_release(%swift.refcounted* %A) ; Instructions which cannot do a release. %L2 = load i64, i64* %C, align 8 store i64 42, i64* %C tail call void @swift_retainUnowned(%swift.refcounted* %A) %L3 = load i64, i64* %addr, align 8 tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @dont_remove_redundant_check_unowned ; CHECK-NEXT: bitcast ; CHECK-NEXT: load ; CHECK-NEXT: call void @swift_checkUnowned ; CHECK-NEXT: call void @unknown_func ; CHECK-NEXT: load ; CHECK-NEXT: call void @swift_checkUnowned ; CHECK-NEXT: ret define void @dont_remove_redundant_check_unowned(%swift.refcounted* %A, %swift.refcounted* %B, i64* %C) { tail call void @swift_retainUnowned(%swift.refcounted* %A) %addr = bitcast %swift.refcounted* %A to i64* %L1 = load i64, i64* %addr, align 8 tail call void @swift_release(%swift.refcounted* %A) ; Could do a release of %A call void @unknown_func() tail call void @swift_retainUnowned(%swift.refcounted* %A) %L3 = load i64, i64* %addr, align 8 tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @unknown_retain_promotion ; CHECK-NEXT: swift_retain ; CHECK-NEXT: swift_retain ; CHECK-NEXT: swift_retain ; CHECK-NEXT: ret define void @unknown_retain_promotion(%swift.refcounted* %A) { tail call void @swift_unknownRetain(%swift.refcounted* %A) tail call void @swift_unknownRetain(%swift.refcounted* %A) tail call void @swift_retain(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @unknown_release_promotion ; CHECK-NEXT: swift_release ; CHECK-NEXT: swift_release ; CHECK-NEXT: swift_release ; CHECK-NEXT: ret define void @unknown_release_promotion(%swift.refcounted* %A) { tail call void @swift_unknownRelease(%swift.refcounted* %A) tail call void @swift_unknownRelease(%swift.refcounted* %A) tail call void @swift_release(%swift.refcounted* %A) ret void } ; CHECK-LABEL: @unknown_retain_nopromotion ; CHECK: bb1 ; CHECK-NOT: swift_retain ; CHECK: ret define void @unknown_retain_nopromotion(%swift.refcounted* %A) { tail call void @swift_retain(%swift.refcounted* %A) br label %bb1 bb1: tail call void @swift_unknownRetain(%swift.refcounted* %A) ret void } !llvm.dbg.cu = !{!1} !llvm.module.flags = !{!4} !0 = !DILocation(line: 0, scope: !3) !1 = distinct !DICompileUnit(language: DW_LANG_Swift, file: !2) !2 = !DIFile(filename: "basic.swift", directory: "") !3 = distinct !DISubprogram(name: "_", scope: !1, file: !2, type: !DISubroutineType(types: !{})) !4 = !{i32 1, !"Debug Info Version", i32 3}