1// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -fblocks -fobjc-arc -fobjc-runtime-has-weak -O2 -disable-llvm-optzns -o - %s | FileCheck %s 2// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -fblocks -fobjc-arc -fobjc-runtime-has-weak -o - %s | FileCheck -check-prefix=CHECK-UNOPT %s 3 4// This shouldn't crash. 5void test0(id (^maker)(void)) { 6 maker(); 7} 8 9int (^test1(int x))(void) { 10 // CHECK-LABEL: define i32 ()* @test1( 11 // CHECK: [[X:%.*]] = alloca i32, 12 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 13 // CHECK-NEXT: store i32 {{%.*}}, i32* [[X]] 14 // CHECK: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to i32 ()* 15 // CHECK-NEXT: [[T1:%.*]] = bitcast i32 ()* [[T0]] to i8* 16 // CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) [[NUW:#[0-9]+]] 17 // CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to i32 ()* 18 // CHECK-NEXT: [[T4:%.*]] = bitcast i32 ()* [[T3]] to i8* 19 // CHECK-NEXT: [[T5:%.*]] = tail call i8* @objc_autoreleaseReturnValue(i8* [[T4]]) [[NUW]] 20 // CHECK-NEXT: [[T6:%.*]] = bitcast i8* [[T5]] to i32 ()* 21 // CHECK-NEXT: ret i32 ()* [[T6]] 22 return ^{ return x; }; 23} 24 25void test2(id x) { 26// CHECK-LABEL: define void @test2( 27// CHECK: [[X:%.*]] = alloca i8*, 28// CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 29// CHECK-NEXT: [[PARM:%.*]] = call i8* @objc_retain(i8* {{%.*}}) 30// CHECK-NEXT: store i8* [[PARM]], i8** [[X]] 31// CHECK-NEXT: [[SLOTREL:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 32// CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 33// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[X]], 34// CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retain(i8* [[T0]]) 35// CHECK-NEXT: store i8* [[T1]], i8** [[SLOT]], 36// CHECK-NEXT: bitcast 37// CHECK-NEXT: call void @test2_helper( 38// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[SLOTREL]] 39// CHECK-NEXT: call void @objc_release(i8* [[T0]]) [[NUW]], !clang.imprecise_release 40// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[X]] 41// CHECK-NEXT: call void @objc_release(i8* [[T0]]) [[NUW]], !clang.imprecise_release 42// CHECK-NEXT: ret void 43 extern void test2_helper(id (^)(void)); 44 test2_helper(^{ return x; }); 45 46// CHECK-LABEL: define internal void @__copy_helper_block_ 47// CHECK: [[T0:%.*]] = load i8*, i8** 48// CHECK-NEXT: [[SRC:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 49// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** 50// CHECK-NEXT: [[DST:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 51// CHECK-NEXT: [[T0:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[SRC]], i32 0, i32 5 52// CHECK-NEXT: [[T1:%.*]] = load i8*, i8** [[T0]] 53// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retain(i8* [[T1]]) [[NUW]] 54// CHECK-NEXT: ret void 55 56// CHECK-LABEL: define internal void @__destroy_helper_block_ 57// CHECK: [[T0:%.*]] = load i8*, i8** 58// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 59// CHECK-NEXT: [[T2:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[T1]], i32 0, i32 5 60// CHECK-NEXT: [[T3:%.*]] = load i8*, i8** [[T2]] 61// CHECK-NEXT: call void @objc_release(i8* [[T3]]) 62// CHECK-NEXT: ret void 63} 64 65void test3(void (^sink)(id*)) { 66 __strong id strong; 67 sink(&strong); 68 69 // CHECK-LABEL: define void @test3( 70 // CHECK: [[SINK:%.*]] = alloca void (i8**)* 71 // CHECK-NEXT: [[STRONG:%.*]] = alloca i8* 72 // CHECK-NEXT: [[TEMP:%.*]] = alloca i8* 73 // CHECK-NEXT: bitcast void (i8**)* {{%.*}} to i8* 74 // CHECK-NEXT: call i8* @objc_retain( 75 // CHECK-NEXT: bitcast i8* 76 // CHECK-NEXT: store void (i8**)* {{%.*}}, void (i8**)** [[SINK]] 77 // CHECK-NEXT: store i8* null, i8** [[STRONG]] 78 79 // CHECK-NEXT: load void (i8**)*, void (i8**)** [[SINK]] 80 // CHECK-NEXT: bitcast 81 // CHECK-NEXT: getelementptr 82 // CHECK-NEXT: [[BLOCK:%.*]] = bitcast 83 // CHECK-NEXT: [[V:%.*]] = load i8*, i8** [[STRONG]] 84 // CHECK-NEXT: store i8* [[V]], i8** [[TEMP]] 85 // CHECK-NEXT: [[F0:%.*]] = load i8*, i8** 86 // CHECK-NEXT: [[F1:%.*]] = bitcast i8* [[F0]] to void (i8*, i8**)* 87 // CHECK-NEXT: call void [[F1]](i8* [[BLOCK]], i8** [[TEMP]]) 88 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[TEMP]] 89 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retain(i8* [[T0]]) 90 // CHECK-NEXT: call void (...) @clang.arc.use(i8* [[V]]) [[NUW]] 91 // CHECK-NEXT: [[T2:%.*]] = load i8*, i8** [[STRONG]] 92 // CHECK-NEXT: store i8* [[T1]], i8** [[STRONG]] 93 // CHECK-NEXT: call void @objc_release(i8* [[T2]]) 94 95 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[STRONG]] 96 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 97 98 // CHECK-NEXT: load void (i8**)*, void (i8**)** [[SINK]] 99 // CHECK-NEXT: bitcast 100 // CHECK-NEXT: call void @objc_release 101 // CHECK-NEXT: ret void 102 103} 104 105void test4(void) { 106 id test4_source(void); 107 void test4_helper(void (^)(void)); 108 __block id var = test4_source(); 109 test4_helper(^{ var = 0; }); 110 111 // CHECK-LABEL: define void @test4() 112 // CHECK: [[VAR:%.*]] = alloca [[BYREF_T:%.*]], 113 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 114 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 2 115 // 0x02000000 - has copy/dispose helpers strong 116 // CHECK-NEXT: store i32 838860800, i32* [[T0]] 117 // CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 6 118 // CHECK-NEXT: [[T0:%.*]] = call i8* @test4_source() 119 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainAutoreleasedReturnValue(i8* [[T0]]) 120 // CHECK-NEXT: store i8* [[T1]], i8** [[SLOT]] 121 // CHECK-NEXT: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 6 122 // 0x42800000 - has signature, copy/dispose helpers, as well as BLOCK_HAS_EXTENDED_LAYOUT 123 // CHECK: store i32 -1040187392, 124 // CHECK: [[T0:%.*]] = bitcast [[BYREF_T]]* [[VAR]] to i8* 125 // CHECK-NEXT: store i8* [[T0]], i8** 126 // CHECK: call void @test4_helper( 127 // CHECK: [[T0:%.*]] = bitcast [[BYREF_T]]* [[VAR]] to i8* 128 // CHECK-NEXT: call void @_Block_object_dispose(i8* [[T0]], i32 8) 129 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[SLOT]] 130 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 131 // CHECK: ret void 132 133 // CHECK-LABEL: define internal void @__Block_byref_object_copy_ 134 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 135 // CHECK-NEXT: load i8*, i8** 136 // CHECK-NEXT: bitcast i8* {{%.*}} to [[BYREF_T]]* 137 // CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 138 // CHECK-NEXT: [[T2:%.*]] = load i8*, i8** [[T1]] 139 // CHECK-NEXT: store i8* [[T2]], i8** [[T0]] 140 // CHECK-NEXT: store i8* null, i8** [[T1]] 141 142 // CHECK-LABEL: define internal void @__Block_byref_object_dispose_ 143 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 144 // CHECK-NEXT: [[T1:%.*]] = load i8*, i8** [[T0]] 145 // CHECK-NEXT: call void @objc_release(i8* [[T1]]) 146 147 // CHECK-LABEL: define internal void @__test4_block_invoke 148 // CHECK: [[SLOT:%.*]] = getelementptr inbounds {{.*}}, i32 0, i32 6 149 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[SLOT]], align 8 150 // CHECK-NEXT: store i8* null, i8** [[SLOT]], 151 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 152 // CHECK-NEXT: ret void 153 154 // CHECK-LABEL: define internal void @__copy_helper_block_ 155 // CHECK: call void @_Block_object_assign(i8* {{%.*}}, i8* {{%.*}}, i32 8) 156 157 // CHECK-LABEL: define internal void @__destroy_helper_block_ 158 // CHECK: call void @_Block_object_dispose(i8* {{%.*}}, i32 8) 159} 160 161void test5(void) { 162 extern id test5_source(void); 163 void test5_helper(void (^)(void)); 164 __unsafe_unretained id var = test5_source(); 165 test5_helper(^{ (void) var; }); 166 167 // CHECK-LABEL: define void @test5() 168 // CHECK: [[VAR:%.*]] = alloca i8* 169 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 170 // CHECK: [[T0:%.*]] = call i8* @test5_source() 171 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainAutoreleasedReturnValue(i8* [[T0]]) 172 // CHECK-NEXT: store i8* [[T1]], i8** [[VAR]], 173 // CHECK-NEXT: call void @objc_release(i8* [[T1]]) 174 // 0x40800000 - has signature but no copy/dispose, as well as BLOCK_HAS_EXTENDED_LAYOUT 175 // CHECK: store i32 -1073741824, i32* 176 // CHECK: [[CAPTURE:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 177 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[VAR]] 178 // CHECK-NEXT: store i8* [[T0]], i8** [[CAPTURE]] 179 // CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to 180 // CHECK: call void @test5_helper 181 // CHECK-NEXT: ret void 182} 183 184void test6(void) { 185 id test6_source(void); 186 void test6_helper(void (^)(void)); 187 __block __weak id var = test6_source(); 188 test6_helper(^{ var = 0; }); 189 190 // CHECK-LABEL: define void @test6() 191 // CHECK: [[VAR:%.*]] = alloca [[BYREF_T:%.*]], 192 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 193 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 2 194 // 0x02000000 - has copy/dispose helpers weak 195 // CHECK-NEXT: store i32 1107296256, i32* [[T0]] 196 // CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 6 197 // CHECK-NEXT: [[T0:%.*]] = call i8* @test6_source() 198 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainAutoreleasedReturnValue(i8* [[T0]]) 199 // CHECK-NEXT: call i8* @objc_initWeak(i8** [[SLOT]], i8* [[T1]]) 200 // CHECK-NEXT: call void @objc_release(i8* [[T1]]) 201 // CHECK-NEXT: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[VAR]], i32 0, i32 6 202 // 0x42800000 - has signature, copy/dispose helpers, as well as BLOCK_HAS_EXTENDED_LAYOUT 203 // CHECK: store i32 -1040187392, 204 // CHECK: [[T0:%.*]] = bitcast [[BYREF_T]]* [[VAR]] to i8* 205 // CHECK-NEXT: store i8* [[T0]], i8** 206 // CHECK: call void @test6_helper( 207 // CHECK: [[T0:%.*]] = bitcast [[BYREF_T]]* [[VAR]] to i8* 208 // CHECK-NEXT: call void @_Block_object_dispose(i8* [[T0]], i32 8) 209 // CHECK-NEXT: call void @objc_destroyWeak(i8** [[SLOT]]) 210 // CHECK: ret void 211 212 // CHECK-LABEL: define internal void @__Block_byref_object_copy_ 213 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 214 // CHECK-NEXT: load i8*, i8** 215 // CHECK-NEXT: bitcast i8* {{%.*}} to [[BYREF_T]]* 216 // CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 217 // CHECK-NEXT: call void @objc_moveWeak(i8** [[T0]], i8** [[T1]]) 218 219 // CHECK-LABEL: define internal void @__Block_byref_object_dispose_ 220 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* {{%.*}}, i32 0, i32 6 221 // CHECK-NEXT: call void @objc_destroyWeak(i8** [[T0]]) 222 223 // CHECK-LABEL: define internal void @__test6_block_invoke 224 // CHECK: [[SLOT:%.*]] = getelementptr inbounds {{.*}}, i32 0, i32 6 225 // CHECK-NEXT: call i8* @objc_storeWeak(i8** [[SLOT]], i8* null) 226 // CHECK-NEXT: ret void 227 228 // CHECK-LABEL: define internal void @__copy_helper_block_ 229 // 0x8 - FIELD_IS_BYREF (no FIELD_IS_WEAK because clang in control) 230 // CHECK: call void @_Block_object_assign(i8* {{%.*}}, i8* {{%.*}}, i32 8) 231 232 // CHECK-LABEL: define internal void @__destroy_helper_block_ 233 // 0x8 - FIELD_IS_BYREF (no FIELD_IS_WEAK because clang in control) 234 // CHECK: call void @_Block_object_dispose(i8* {{%.*}}, i32 8) 235} 236 237void test7(void) { 238 id test7_source(void); 239 void test7_helper(void (^)(void)); 240 void test7_consume(id); 241 __weak id var = test7_source(); 242 test7_helper(^{ test7_consume(var); }); 243 244 // CHECK-LABEL: define void @test7() 245 // CHECK: [[VAR:%.*]] = alloca i8*, 246 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 247 // CHECK: [[T0:%.*]] = call i8* @test7_source() 248 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainAutoreleasedReturnValue(i8* [[T0]]) 249 // CHECK-NEXT: call i8* @objc_initWeak(i8** [[VAR]], i8* [[T1]]) 250 // CHECK-NEXT: call void @objc_release(i8* [[T1]]) 251 // 0x42800000 - has signature, copy/dispose helpers, as well as BLOCK_HAS_EXTENDED_LAYOUT 252 // CHECK: store i32 -1040187392, 253 // CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 254 // CHECK-NEXT: [[T0:%.*]] = call i8* @objc_loadWeakRetained(i8** [[VAR]]) 255 // CHECK-NEXT: call i8* @objc_initWeak(i8** [[SLOT]], i8* [[T0]]) 256 // CHECK: call void @test7_helper( 257 // CHECK-NEXT: call void @objc_destroyWeak(i8** {{%.*}}) 258 // CHECK-NEXT: call void @objc_destroyWeak(i8** [[VAR]]) 259 // CHECK: ret void 260 261 // CHECK-LABEL: define internal void @__test7_block_invoke 262 // CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* {{%.*}}, i32 0, i32 5 263 // CHECK-NEXT: [[T0:%.*]] = call i8* @objc_loadWeakRetained(i8** [[SLOT]]) 264 // CHECK-NEXT: call void @test7_consume(i8* [[T0]]) 265 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 266 // CHECK: ret void 267 268 // CHECK-LABEL: define internal void @__copy_helper_block_ 269 // CHECK: getelementptr 270 // CHECK-NEXT: getelementptr 271 // CHECK-NEXT: call void @objc_copyWeak( 272 273 // CHECK-LABEL: define internal void @__destroy_helper_block_ 274 // CHECK: getelementptr 275 // CHECK-NEXT: call void @objc_destroyWeak( 276} 277 278@interface Test8 @end 279@implementation Test8 280- (void) test { 281// CHECK: define internal void @"\01-[Test8 test]" 282// CHECK: [[SELF:%.*]] = alloca [[TEST8:%.*]]*, 283// CHECK-NEXT: alloca i8* 284// CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 285// CHECK: store 286// CHECK-NEXT: store 287// CHECK: [[D0:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 288// CHECK: [[T0:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 289// CHECK-NEXT: [[T1:%.*]] = load [[TEST8]]*, [[TEST8]]** [[SELF]], 290// CHECK-NEXT: [[T2:%.*]] = bitcast [[TEST8]]* [[T1]] to i8* 291// CHECK-NEXT: [[T3:%.*]] = call i8* @objc_retain(i8* [[T2]]) 292// CHECK-NEXT: [[T4:%.*]] = bitcast i8* [[T3]] to [[TEST8]]* 293// CHECK-NEXT: store [[TEST8]]* [[T4]], [[TEST8]]** [[T0]] 294// CHECK-NEXT: bitcast [[BLOCK_T]]* [[BLOCK]] to 295// CHECK: call void @test8_helper( 296// CHECK-NEXT: [[T1:%.*]] = load [[TEST8]]*, [[TEST8]]** [[D0]] 297// CHECK-NEXT: [[T2:%.*]] = bitcast [[TEST8]]* [[T1]] to i8* 298// CHECK-NEXT: call void @objc_release(i8* [[T2]]) 299// CHECK: ret void 300 301 extern void test8_helper(void (^)(void)); 302 test8_helper(^{ (void) self; }); 303} 304@end 305 306id test9(void) { 307 typedef id __attribute__((ns_returns_retained)) blocktype(void); 308 extern void test9_consume_block(blocktype^); 309 return ^blocktype { 310 extern id test9_produce(void); 311 return test9_produce(); 312 }(); 313 314// CHECK-LABEL: define i8* @test9( 315// CHECK: load i8*, i8** getelementptr 316// CHECK-NEXT: bitcast i8* 317// CHECK-NEXT: call i8* 318// CHECK-NEXT: tail call i8* @objc_autoreleaseReturnValue 319// CHECK-NEXT: ret i8* 320 321// CHECK: call i8* @test9_produce() 322// CHECK-NEXT: call i8* @objc_retain 323// CHECK-NEXT: ret i8* 324} 325 326// rdar://problem/9814099 327// Test that we correctly initialize __block variables 328// when the initialization captures the variable. 329void test10a(void) { 330 __block void (^block)(void) = ^{ block(); }; 331 // CHECK-LABEL: define void @test10a() 332 // CHECK: [[BYREF:%.*]] = alloca [[BYREF_T:%.*]], 333 334 // Zero-initialization before running the initializer. 335 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 6 336 // CHECK-NEXT: store void ()* null, void ()** [[T0]], align 8 337 338 // Run the initializer as an assignment. 339 // CHECK: [[T0:%.*]] = bitcast void ()* {{%.*}} to i8* 340 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainBlock(i8* [[T0]]) 341 // CHECK-NEXT: [[T2:%.*]] = bitcast i8* [[T1]] to void ()* 342 // CHECK-NEXT: [[T3:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 1 343 // CHECK-NEXT: [[T4:%.*]] = load [[BYREF_T]]*, [[BYREF_T]]** [[T3]] 344 // CHECK-NEXT: [[T5:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[T4]], i32 0, i32 6 345 // CHECK-NEXT: [[T6:%.*]] = load void ()*, void ()** [[T5]], align 8 346 // CHECK-NEXT: store void ()* {{%.*}}, void ()** [[T5]], align 8 347 // CHECK-NEXT: [[T7:%.*]] = bitcast void ()* [[T6]] to i8* 348 // CHECK-NEXT: call void @objc_release(i8* [[T7]]) 349 350 // Destroy at end of function. 351 // CHECK-NEXT: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 6 352 // CHECK-NEXT: [[T0:%.*]] = bitcast [[BYREF_T]]* [[BYREF]] to i8* 353 // CHECK-NEXT: call void @_Block_object_dispose(i8* [[T0]], i32 8) 354 // CHECK-NEXT: [[T1:%.*]] = load void ()*, void ()** [[SLOT]] 355 // CHECK-NEXT: [[T2:%.*]] = bitcast void ()* [[T1]] to i8* 356 // CHECK-NEXT: call void @objc_release(i8* [[T2]]) 357 // CHECK: ret void 358} 359 360// <rdar://problem/10402698>: do this copy and dispose with 361// objc_retainBlock/release instead of _Block_object_assign/destroy. 362// We can also use _Block_object_assign/destroy with 363// BLOCK_FIELD_IS_BLOCK as long as we don't pass BLOCK_BYREF_CALLER. 364 365// CHECK-LABEL: define internal void @__Block_byref_object_copy 366// CHECK: [[D0:%.*]] = load i8*, i8** {{%.*}} 367// CHECK-NEXT: [[D1:%.*]] = bitcast i8* [[D0]] to [[BYREF_T]]* 368// CHECK-NEXT: [[D2:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[D1]], i32 0, i32 6 369// CHECK-NEXT: [[S0:%.*]] = load i8*, i8** {{%.*}} 370// CHECK-NEXT: [[S1:%.*]] = bitcast i8* [[S0]] to [[BYREF_T]]* 371// CHECK-NEXT: [[S2:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[S1]], i32 0, i32 6 372// CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[S2]], align 8 373// CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 374// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 375// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to void ()* 376// CHECK-NEXT: store void ()* [[T3]], void ()** [[D2]], align 8 377// CHECK: ret void 378 379// CHECK-LABEL: define internal void @__Block_byref_object_dispose 380// CHECK: [[T0:%.*]] = load i8*, i8** {{%.*}} 381// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[BYREF_T]]* 382// CHECK-NEXT: [[T2:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[T1]], i32 0, i32 6 383// CHECK-NEXT: [[T3:%.*]] = load void ()*, void ()** [[T2]] 384// CHECK-NEXT: [[T4:%.*]] = bitcast void ()* [[T3]] to i8* 385// CHECK-NEXT: call void @objc_release(i8* [[T4]]) 386// CHECK-NEXT: ret void 387 388// Test that we correctly assign to __block variables when the 389// assignment captures the variable. 390void test10b(void) { 391 __block void (^block)(void); 392 block = ^{ block(); }; 393 394 // CHECK-LABEL: define void @test10b() 395 // CHECK: [[BYREF:%.*]] = alloca [[BYREF_T:%.*]], 396 397 // Zero-initialize. 398 // CHECK: [[T0:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 6 399 // CHECK-NEXT: store void ()* null, void ()** [[T0]], align 8 400 401 // CHECK-NEXT: [[SLOT:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 6 402 403 // The assignment. 404 // CHECK: [[T0:%.*]] = bitcast void ()* {{%.*}} to i8* 405 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainBlock(i8* [[T0]]) 406 // CHECK-NEXT: [[T2:%.*]] = bitcast i8* [[T1]] to void ()* 407 // CHECK-NEXT: [[T3:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[BYREF]], i32 0, i32 1 408 // CHECK-NEXT: [[T4:%.*]] = load [[BYREF_T]]*, [[BYREF_T]]** [[T3]] 409 // CHECK-NEXT: [[T5:%.*]] = getelementptr inbounds [[BYREF_T]], [[BYREF_T]]* [[T4]], i32 0, i32 6 410 // CHECK-NEXT: [[T6:%.*]] = load void ()*, void ()** [[T5]], align 8 411 // CHECK-NEXT: store void ()* {{%.*}}, void ()** [[T5]], align 8 412 // CHECK-NEXT: [[T7:%.*]] = bitcast void ()* [[T6]] to i8* 413 // CHECK-NEXT: call void @objc_release(i8* [[T7]]) 414 415 // Destroy at end of function. 416 // CHECK-NEXT: [[T0:%.*]] = bitcast [[BYREF_T]]* [[BYREF]] to i8* 417 // CHECK-NEXT: call void @_Block_object_dispose(i8* [[T0]], i32 8) 418 // CHECK-NEXT: [[T1:%.*]] = load void ()*, void ()** [[SLOT]] 419 // CHECK-NEXT: [[T2:%.*]] = bitcast void ()* [[T1]] to i8* 420 // CHECK-NEXT: call void @objc_release(i8* [[T2]]) 421 // CHECK: ret void 422} 423 424// rdar://problem/10088932 425void test11_helper(id); 426void test11a(void) { 427 int x; 428 test11_helper(^{ (void) x; }); 429 430 // CHECK-LABEL: define void @test11a() 431 // CHECK: [[X:%.*]] = alloca i32, align 4 432 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], align 8 433 // CHECK: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to void ()* 434 // CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 435 // CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 436 // CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to void ()* 437 // CHECK-NEXT: [[T4:%.*]] = bitcast void ()* [[T3]] to i8* 438 // CHECK-NEXT: call void @test11_helper(i8* [[T4]]) 439 // CHECK-NEXT: [[T5:%.*]] = bitcast void ()* [[T3]] to i8* 440 // CHECK-NEXT: call void @objc_release(i8* [[T5]]) 441 // CHECK: ret void 442} 443void test11b(void) { 444 int x; 445 id b = ^{ (void) x; }; 446 447 // CHECK-LABEL: define void @test11b() 448 // CHECK: [[X:%.*]] = alloca i32, align 4 449 // CHECK-NEXT: [[B:%.*]] = alloca i8*, align 8 450 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], align 8 451 // CHECK: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to void ()* 452 // CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 453 // CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 454 // CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to void ()* 455 // CHECK-NEXT: [[T4:%.*]] = bitcast void ()* [[T3]] to i8* 456 // CHECK-NEXT: store i8* [[T4]], i8** [[B]], align 8 457 // CHECK-NEXT: [[T5:%.*]] = load i8*, i8** [[B]] 458 // CHECK-NEXT: call void @objc_release(i8* [[T5]]) 459 // CHECK: ret void 460} 461 462// rdar://problem/9979150 463@interface Test12 464@property (strong) void(^ablock)(void); 465@property (nonatomic, strong) void(^nblock)(void); 466@end 467@implementation Test12 468@synthesize ablock, nblock; 469// CHECK: define internal void ()* @"\01-[Test12 ablock]"( 470// CHECK: call i8* @objc_getProperty(i8* {{%.*}}, i8* {{%.*}}, i64 {{%.*}}, i1 zeroext true) 471 472// CHECK: define internal void @"\01-[Test12 setAblock:]"( 473// CHECK: call void @objc_setProperty(i8* {{%.*}}, i8* {{%.*}}, i64 {{%.*}}, i8* {{%.*}}, i1 zeroext true, i1 zeroext true) 474 475// CHECK: define internal void ()* @"\01-[Test12 nblock]"( 476// CHECK: call i8* @objc_getProperty(i8* {{%.*}}, i8* {{%.*}}, i64 {{%.*}}, i1 zeroext false) 477 478// CHECK: define internal void @"\01-[Test12 setNblock:]"( 479// CHECK: call void @objc_setProperty(i8* {{%.*}}, i8* {{%.*}}, i64 {{%.*}}, i8* {{%.*}}, i1 zeroext false, i1 zeroext true) 480@end 481 482// rdar://problem/10131784 483void test13(id x) { 484 extern void test13_helper(id); 485 extern void test13_use(void(^)(void)); 486 487 void (^b)(void) = (x ? ^{test13_helper(x);} : 0); 488 test13_use(b); 489 490 // CHECK-LABEL: define void @test13( 491 // CHECK: [[X:%.*]] = alloca i8*, align 8 492 // CHECK-NEXT: [[B:%.*]] = alloca void ()*, align 8 493 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:.*]], align 8 494 // CHECK-NEXT: [[CLEANUP_ACTIVE:%.*]] = alloca i1 495 // CHECK-NEXT: [[T0:%.*]] = call i8* @objc_retain(i8* {{%.*}}) 496 // CHECK-NEXT: store i8* [[T0]], i8** [[X]], align 8 497 // CHECK-NEXT: [[CLEANUP_ADDR:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 498 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[X]], align 8 499 // CHECK-NEXT: [[T1:%.*]] = icmp ne i8* [[T0]], null 500 // CHECK-NEXT: store i1 false, i1* [[CLEANUP_ACTIVE]] 501 // CHECK-NEXT: br i1 [[T1]], 502 503 // CHECK-NOT: br 504 // CHECK: [[CAPTURE:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 505 // CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[X]], align 8 506 // CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retain(i8* [[T0]]) 507 // CHECK-NEXT: store i8* [[T1]], i8** [[CAPTURE]], align 8 508 // CHECK-NEXT: store i1 true, i1* [[CLEANUP_ACTIVE]] 509 // CHECK-NEXT: bitcast [[BLOCK_T]]* [[BLOCK]] to void ()* 510 // CHECK-NEXT: br label 511 // CHECK: br label 512 // CHECK: [[T0:%.*]] = phi void ()* 513 // CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 514 // CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 515 // CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to void ()* 516 // CHECK-NEXT: store void ()* [[T3]], void ()** [[B]], align 8 517 // CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[B]], align 8 518 // CHECK-NEXT: call void @test13_use(void ()* [[T0]]) 519 // CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[B]] 520 // CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 521 // CHECK-NEXT: call void @objc_release(i8* [[T1]]) 522 523 // CHECK-NEXT: [[T0:%.*]] = load i1, i1* [[CLEANUP_ACTIVE]] 524 // CHECK-NEXT: br i1 [[T0]] 525 // CHECK: [[T0:%.*]] = load i8*, i8** [[CLEANUP_ADDR]] 526 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 527 // CHECK-NEXT: br label 528 529 // CHECK: [[T0:%.*]] = load i8*, i8** [[X]] 530 // CHECK-NEXT: call void @objc_release(i8* [[T0]]) 531 // CHECK-NEXT: ret void 532} 533 534// <rdar://problem/10907510> 535void test14() { 536 void (^const x[1])(void) = { ^{} }; 537} 538 539// rdar://11149025 540// Don't make invalid ASTs and crash. 541void test15_helper(void (^block)(void), int x); 542void test15(int a) { 543 test15_helper(^{ (void) a; }, ({ a; })); 544} 545 546// rdar://11016025 547void test16() { 548 void (^BLKVAR)(void) = ^{ BLKVAR(); }; 549 550 // CHECK-LABEL: define void @test16( 551 // CHECK: [[BLKVAR:%.*]] = alloca void ()*, align 8 552 // CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 553 // CHECK-NEXT: [[SLOTREL:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 554 // CHECK-NEXT: store void ()* null, void ()** [[BLKVAR]], align 8 555} 556 557// rdar://12151005 558// 559// This is an intentional exception to our conservative jump-scope 560// checking for full-expressions containing block literals with 561// non-trivial cleanups: if the block literal appears in the operand 562// of a return statement, there's no need to extend its lifetime. 563id (^test17(id self, int which))(void) { 564 switch (which) { 565 case 1: return ^{ return self; }; 566 case 0: return ^{ return self; }; 567 } 568 return (void*) 0; 569} 570// CHECK-LABEL: define i8* ()* @test17( 571// CHECK: [[RET:%.*]] = alloca i8* ()*, align 572// CHECK-NEXT: [[SELF:%.*]] = alloca i8*, 573// CHECK: [[B0:%.*]] = alloca [[BLOCK:<.*>]], align 574// CHECK: [[B1:%.*]] = alloca [[BLOCK]], align 575// CHECK: [[T0:%.*]] = call i8* @objc_retain(i8* 576// CHECK-NEXT: store i8* [[T0]], i8** [[SELF]], align 577// CHECK-NOT: objc_retain 578// CHECK-NOT: objc_release 579// CHECK: [[DESTROY:%.*]] = getelementptr inbounds [[BLOCK]], [[BLOCK]]* [[B0]], i32 0, i32 5 580// CHECK-NOT: objc_retain 581// CHECK-NOT: objc_release 582// CHECK: [[T0:%.*]] = getelementptr inbounds [[BLOCK]], [[BLOCK]]* [[B0]], i32 0, i32 5 583// CHECK-NEXT: [[T1:%.*]] = load i8*, i8** [[SELF]], align 584// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retain(i8* [[T1]]) 585// CHECK-NEXT: store i8* [[T2]], i8** [[T0]], 586// CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK]]* [[B0]] to i8* ()* 587// CHECK-NEXT: [[T1:%.*]] = bitcast i8* ()* [[T0]] to i8* 588// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 589// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to i8* ()* 590// CHECK-NEXT: store i8* ()* [[T3]], i8* ()** [[RET]] 591// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[DESTROY]] 592// CHECK-NEXT: call void @objc_release(i8* [[T0]]) 593// CHECK-NEXT: store i32 594// CHECK-NEXT: br label 595// CHECK-NOT: objc_retain 596// CHECK-NOT: objc_release 597// CHECK: [[DESTROY:%.*]] = getelementptr inbounds [[BLOCK]], [[BLOCK]]* [[B1]], i32 0, i32 5 598// CHECK-NOT: objc_retain 599// CHECK-NOT: objc_release 600// CHECK: [[T0:%.*]] = getelementptr inbounds [[BLOCK]], [[BLOCK]]* [[B1]], i32 0, i32 5 601// CHECK-NEXT: [[T1:%.*]] = load i8*, i8** [[SELF]], align 602// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retain(i8* [[T1]]) 603// CHECK-NEXT: store i8* [[T2]], i8** [[T0]], 604// CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK]]* [[B1]] to i8* ()* 605// CHECK-NEXT: [[T1:%.*]] = bitcast i8* ()* [[T0]] to i8* 606// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retainBlock(i8* [[T1]]) 607// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to i8* ()* 608// CHECK-NEXT: store i8* ()* [[T3]], i8* ()** [[RET]] 609// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[DESTROY]] 610// CHECK-NEXT: call void @objc_release(i8* [[T0]]) 611// CHECK-NEXT: store i32 612// CHECK-NEXT: br label 613 614void test18(id x) { 615// CHECK-UNOPT-LABEL: define void @test18( 616// CHECK-UNOPT: [[X:%.*]] = alloca i8*, 617// CHECK-UNOPT-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 618// CHECK-UNOPT-NEXT: store i8* null, i8** [[X]] 619// CHECK-UNOPT-NEXT: call void @objc_storeStrong(i8** [[X]], 620// CHECK-UNOPT-NEXT: [[SLOTREL:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 621// CHECK-UNOPT: [[SLOT:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 622// CHECK-UNOPT-NEXT: [[T0:%.*]] = load i8*, i8** [[X]], 623// CHECK-UNOPT-NEXT: [[T1:%.*]] = call i8* @objc_retain(i8* [[T0]]) 624// CHECK-UNOPT-NEXT: store i8* [[T1]], i8** [[SLOT]], 625// CHECK-UNOPT-NEXT: bitcast 626// CHECK-UNOPT-NEXT: call void @test18_helper( 627// CHECK-UNOPT-NEXT: call void @objc_storeStrong(i8** [[SLOTREL]], i8* null) [[NUW:#[0-9]+]] 628// CHECK-UNOPT-NEXT: call void @objc_storeStrong(i8** [[X]], i8* null) [[NUW]] 629// CHECK-UNOPT-NEXT: ret void 630 extern void test18_helper(id (^)(void)); 631 test18_helper(^{ return x; }); 632 633// CHECK-UNOPT-LABEL: define internal void @__copy_helper_block_ 634// CHECK-UNOPT: [[T0:%.*]] = load i8*, i8** 635// CHECK-UNOPT-NEXT: [[SRC:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 636// CHECK-UNOPT-NEXT: [[T0:%.*]] = load i8*, i8** 637// CHECK-UNOPT-NEXT: [[DST:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 638// CHECK-UNOPT-NEXT: [[T0:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[SRC]], i32 0, i32 5 639// CHECK-UNOPT-NEXT: [[T1:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[DST]], i32 0, i32 5 640// CHECK-UNOPT-NEXT: [[T2:%.*]] = load i8*, i8** [[T0]] 641// CHECK-UNOPT-NEXT: store i8* null, i8** [[T1]] 642// CHECK-UNOPT-NEXT: call void @objc_storeStrong(i8** [[T1]], i8* [[T2]]) [[NUW]] 643// CHECK-UNOPT-NEXT: ret void 644 645// CHECK-UNOPT-LABEL: define internal void @__destroy_helper_block_ 646// CHECK-UNOPT: [[T0:%.*]] = load i8*, i8** 647// CHECK-UNOPT-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[BLOCK_T]]* 648// CHECK-UNOPT-NEXT: [[T2:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[T1]], i32 0, i32 5 649// CHECK-UNOPT-NEXT: call void @objc_storeStrong(i8** [[T2]], i8* null) 650// CHECK-UNOPT-NEXT: ret void 651} 652 653// rdar://13588325 654void test19_sink(void (^)(int)); 655void test19(void (^b)(void)) { 656// CHECK-LABEL: define void @test19( 657// Prologue. 658// CHECK: [[B:%.*]] = alloca void ()*, 659// CHECK-NEXT: [[BLOCK:%.*]] = alloca [[BLOCK_T:<{.*}>]], 660// CHECK-NEXT: [[T0:%.*]] = bitcast void ()* {{%.*}} to i8* 661// CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retain(i8* [[T0]]) 662// CHECK-NEXT: [[T2:%.*]] = bitcast i8* [[T1]] to void ()* 663// CHECK-NEXT: store void ()* [[T2]], void ()** [[B]] 664 665// Block setup. We skip most of this. Note the bare retain. 666// CHECK-NEXT: [[SLOTREL:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 667// CHECK: [[SLOT:%.*]] = getelementptr inbounds [[BLOCK_T]], [[BLOCK_T]]* [[BLOCK]], i32 0, i32 5 668// CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[B]], 669// CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 670// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_retain(i8* [[T1]]) 671// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to void ()* 672// CHECK-NEXT: store void ()* [[T3]], void ()** [[SLOT]], 673// Call. 674// CHECK-NEXT: [[T0:%.*]] = bitcast [[BLOCK_T]]* [[BLOCK]] to void (i32)* 675// CHECK-NEXT: call void @test19_sink(void (i32)* [[T0]]) 676 677 test19_sink(^(int x) { b(); }); 678 679// Block teardown. 680// CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[SLOTREL]] 681// CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 682// CHECK-NEXT: call void @objc_release(i8* [[T1]]) 683 684// Local cleanup. 685// CHECK-NEXT: [[T0:%.*]] = load void ()*, void ()** [[B]] 686// CHECK-NEXT: [[T1:%.*]] = bitcast void ()* [[T0]] to i8* 687// CHECK-NEXT: call void @objc_release(i8* [[T1]]) 688 689// CHECK-NEXT: ret void 690} 691 692// CHECK: attributes [[NUW]] = { nounwind } 693// CHECK-UNOPT: attributes [[NUW]] = { nounwind } 694