[clang][modules] Don't prevent translation of FW_Private includes when explicitly...
[llvm-project.git] / llvm / test / Transforms / ObjCARC / invoke.ll
bloba792b8e8683668f51a1478a8fbd32a6712bd4fdf
1 ; RUN: opt -S -passes=objc-arc < %s | FileCheck %s
3 declare ptr @llvm.objc.retain(ptr)
4 declare void @llvm.objc.release(ptr)
5 declare ptr @llvm.objc.retainAutoreleasedReturnValue(ptr)
6 declare ptr @objc_msgSend(ptr, ptr, ...)
7 declare void @use_pointer(ptr)
8 declare void @callee()
9 declare ptr @returner()
11 ; ARCOpt shouldn't try to move the releases to the block containing the invoke.
13 ; CHECK-LABEL: define void @test0(
14 ; CHECK: invoke.cont:
15 ; CHECK:   call void @llvm.objc.release(ptr %zipFile) [[NUW:#[0-9]+]], !clang.imprecise_release !0
16 ; CHECK:   ret void
17 ; CHECK: lpad:
18 ; CHECK:   call void @llvm.objc.release(ptr %zipFile) [[NUW]], !clang.imprecise_release !0
19 ; CHECK:   ret void
20 ; CHECK-NEXT: }
21 define void @test0(ptr %zipFile) personality ptr @__gxx_personality_v0 {
22 entry:
23   call ptr @llvm.objc.retain(ptr %zipFile) nounwind
24   call void @use_pointer(ptr %zipFile)
25   invoke void @objc_msgSend(ptr %zipFile) 
26           to label %invoke.cont unwind label %lpad
28 invoke.cont:                                      ; preds = %entry
29   call void @llvm.objc.release(ptr %zipFile) nounwind, !clang.imprecise_release !0
30   ret void
32 lpad:                                             ; preds = %entry
33   %exn = landingpad {ptr, i32}
34            cleanup
35   call void @llvm.objc.release(ptr %zipFile) nounwind, !clang.imprecise_release !0
36   ret void
39 ; ARCOpt should move the release before the callee calls.
41 ; CHECK-LABEL: define void @test1(
42 ; CHECK: invoke.cont:
43 ; CHECK:   call void @llvm.objc.release(ptr %zipFile) [[NUW]], !clang.imprecise_release !0
44 ; CHECK:   call void @callee()
45 ; CHECK:   br label %done
46 ; CHECK: lpad:
47 ; CHECK:   call void @llvm.objc.release(ptr %zipFile) [[NUW]], !clang.imprecise_release !0
48 ; CHECK:   call void @callee()
49 ; CHECK:   br label %done
50 ; CHECK: done:
51 ; CHECK-NEXT: ret void
52 ; CHECK-NEXT: }
53 define void @test1(ptr %zipFile) personality ptr @__gxx_personality_v0 {
54 entry:
55   call ptr @llvm.objc.retain(ptr %zipFile) nounwind
56   call void @use_pointer(ptr %zipFile)
57   invoke void @objc_msgSend(ptr %zipFile)
58           to label %invoke.cont unwind label %lpad
60 invoke.cont:                                      ; preds = %entry
61   call void @callee()
62   br label %done
64 lpad:                                             ; preds = %entry
65   %exn = landingpad {ptr, i32}
66            cleanup
67   call void @callee()
68   br label %done
70 done:
71   call void @llvm.objc.release(ptr %zipFile) nounwind, !clang.imprecise_release !0
72   ret void
75 ; The optimizer should ignore invoke unwind paths consistently.
76 ; PR12265
78 ; CHECK: define void @test2() personality ptr @__objc_personality_v0 {
79 ; CHECK: invoke.cont:
80 ; CHECK-NEXT: call ptr @llvm.objc.retain
81 ; CHECK-NOT: @llvm.objc.r
82 ; CHECK: finally.cont:
83 ; CHECK-NEXT: call void @llvm.objc.release
84 ; CHECK-NOT: @objc
85 ; CHECK: finally.rethrow:
86 ; CHECK-NOT: @objc
87 ; CHECK: }
88 define void @test2() personality ptr @__objc_personality_v0 {
89 entry:
90   %call = invoke ptr @objc_msgSend()
91           to label %invoke.cont unwind label %finally.rethrow, !clang.arc.no_objc_arc_exceptions !0
93 invoke.cont:                                      ; preds = %entry
94   %tmp1 = tail call ptr @llvm.objc.retainAutoreleasedReturnValue(ptr %call) nounwind
95   call void @objc_msgSend(), !clang.arc.no_objc_arc_exceptions !0
96   invoke void @use_pointer(ptr %call)
97           to label %finally.cont unwind label %finally.rethrow, !clang.arc.no_objc_arc_exceptions !0
99 finally.cont:                                     ; preds = %invoke.cont
100   tail call void @llvm.objc.release(ptr %call) nounwind, !clang.imprecise_release !0
101   ret void
103 finally.rethrow:                                  ; preds = %invoke.cont, %entry
104   %tmp2 = landingpad { ptr, i32 }
105           catch ptr null
106   unreachable
109 ; Don't try to place code on invoke critical edges.
111 ; CHECK-LABEL: define void @test3(
112 ; CHECK: if.end:
113 ; CHECK-NEXT: call void @llvm.objc.release(ptr %p) [[NUW]]
114 ; CHECK-NEXT: ret void
115 ; CHECK-NEXT: }
116 define void @test3(ptr %p, i1 %b) personality ptr @__objc_personality_v0 {
117 entry:
118   %0 = call ptr @llvm.objc.retain(ptr %p)
119   call void @callee()
120   br i1 %b, label %if.else, label %if.then
122 if.then:
123   invoke void @use_pointer(ptr %p)
124           to label %if.end unwind label %lpad, !clang.arc.no_objc_arc_exceptions !0
126 if.else:
127   invoke void @use_pointer(ptr %p)
128           to label %if.end unwind label %lpad, !clang.arc.no_objc_arc_exceptions !0
130 lpad:
131   %r = landingpad { ptr, i32 }
132        cleanup
133   ret void
135 if.end:
136   call void @llvm.objc.release(ptr %p)
137   ret void
140 ; Like test3, but with ARC-relevant exception handling.
142 ; CHECK-LABEL: define void @test4(
143 ; CHECK: lpad:
144 ; CHECK-NEXT: %r = landingpad { ptr, i32 }
145 ; CHECK-NEXT: cleanup
146 ; CHECK-NEXT: call void @llvm.objc.release(ptr %p) [[NUW]]
147 ; CHECK-NEXT: ret void
148 ; CHECK: if.end:
149 ; CHECK-NEXT: call void @llvm.objc.release(ptr %p) [[NUW]]
150 ; CHECK-NEXT: ret void
151 ; CHECK-NEXT: }
152 define void @test4(ptr %p, i1 %b) personality ptr @__objc_personality_v0 {
153 entry:
154   %0 = call ptr @llvm.objc.retain(ptr %p)
155   call void @callee()
156   br i1 %b, label %if.else, label %if.then
158 if.then:
159   invoke void @use_pointer(ptr %p)
160           to label %if.end unwind label %lpad
162 if.else:
163   invoke void @use_pointer(ptr %p)
164           to label %if.end unwind label %lpad
166 lpad:
167   %r = landingpad { ptr, i32 }
168        cleanup
169   call void @llvm.objc.release(ptr %p)
170   ret void
172 if.end:
173   call void @llvm.objc.release(ptr %p)
174   ret void
177 ; Don't turn the retainAutoreleaseReturnValue into retain, because it's
178 ; for an invoke which we can assume codegen will put immediately prior.
180 ; CHECK-LABEL: define void @test5(
181 ; CHECK: call ptr @llvm.objc.retainAutoreleasedReturnValue(ptr %z)
182 ; CHECK: }
183 define void @test5() personality ptr @__objc_personality_v0 {
184 entry:
185   %z = invoke ptr @returner()
186           to label %if.end unwind label %lpad, !clang.arc.no_objc_arc_exceptions !0
188 lpad:
189   %r13 = landingpad { ptr, i32 }
190           cleanup
191   ret void
193 if.end:
194   call ptr @llvm.objc.retainAutoreleasedReturnValue(ptr %z)
195   ret void
198 ; Like test5, but there's intervening code.
200 ; CHECK-LABEL: define void @test6(
201 ; CHECK: call ptr @llvm.objc.retain(ptr %z)
202 ; CHECK: }
203 define void @test6() personality ptr @__objc_personality_v0 {
204 entry:
205   %z = invoke ptr @returner()
206           to label %if.end unwind label %lpad, !clang.arc.no_objc_arc_exceptions !0
208 lpad:
209   %r13 = landingpad { ptr, i32 }
210           cleanup
211   ret void
213 if.end:
214   call void @callee()
215   call ptr @llvm.objc.retainAutoreleasedReturnValue(ptr %z)
216   ret void
219 declare i32 @__gxx_personality_v0(...)
220 declare i32 @__objc_personality_v0(...)
222 ; CHECK: attributes [[NUW]] = { nounwind }
224 !0 = !{}