| 1 | // RUN: %clang_cc1 -std=c++11 -triple x86_64-apple-darwin10 -emit-llvm -fobjc-runtime-has-weak -fblocks -fobjc-arc -O2 -disable-llvm-passes -o - %s | FileCheck %s |
| 2 | |
| 3 | @interface A |
| 4 | @end |
| 5 | |
| 6 | id getObject(); |
| 7 | void callee(); |
| 8 | |
| 9 | // Lifetime extension for binding a reference to an rvalue |
| 10 | // CHECK-LABEL: define void @_Z5test0v() |
| 11 | void test0() { |
| 12 | // CHECK: call i8* @_Z9getObjectv |
| 13 | // CHECK-NEXT: call i8* @llvm.objc.retainAutoreleasedReturnValue |
| 14 | const __strong id &ref1 = getObject(); |
| 15 | // CHECK: call void @_Z6calleev |
| 16 | callee(); |
| 17 | // CHECK: call i8* @_Z9getObjectv |
| 18 | // CHECK-NEXT: call i8* @llvm.objc.retainAutoreleasedReturnValue |
| 19 | // CHECK-NEXT: call i8* @llvm.objc.autorelease |
| 20 | const __autoreleasing id &ref2 = getObject(); |
| 21 | // CHECK: call void @_Z6calleev |
| 22 | callee(); |
| 23 | // CHECK: call void @llvm.objc.release |
| 24 | // CHECK: ret |
| 25 | } |
| 26 | |
| 27 | // No lifetime extension when we're binding a reference to an lvalue. |
| 28 | // CHECK-LABEL: define void @_Z5test1RU8__strongP11objc_objectRU6__weakS0_ |
| 29 | void test1(__strong id &x, __weak id &y) { |
| 30 | // CHECK-NOT: release |
| 31 | const __strong id &ref1 = x; |
| 32 | const __autoreleasing id &ref2 = x; |
| 33 | const __weak id &ref3 = y; |
| 34 | // CHECK: ret void |
| 35 | } |
| 36 | |
| 37 | typedef __strong id strong_id; |
| 38 | |
| 39 | //CHECK: define void @_Z5test3v |
| 40 | void test3() { |
| 41 | // CHECK: [[REF:%.*]] = alloca i8**, align 8 |
| 42 | // CHECK: call i8* @llvm.objc.initWeak |
| 43 | // CHECK-NEXT: store i8** |
| 44 | const __weak id &ref = strong_id(); |
| 45 | // CHECK-NEXT: call void @_Z6calleev() |
| 46 | callee(); |
| 47 | // CHECK-NEXT: call void @llvm.objc.destroyWeak |
| 48 | // CHECK-NEXT: [[PTR:%.*]] = bitcast i8*** [[REF]] to i8* |
| 49 | // CHECK-NEXT: call void @llvm.lifetime.end.p0i8(i64 8, i8* [[PTR]]) |
| 50 | // CHECK-NEXT: ret void |
| 51 | } |
| 52 | |
| 53 | // CHECK-LABEL: define void @_Z5test4RU8__strongP11objc_object |
| 54 | void test4(__strong id &x) { |
| 55 | // CHECK: call i8* @llvm.objc.retain |
| 56 | __strong A* const &ar = x; |
| 57 | // CHECK: store i32 17, i32* |
| 58 | int i = 17; |
| 59 | // CHECK: call void @llvm.objc.release( |
| 60 | // CHECK: ret void |
| 61 | } |
| 62 | |
| 63 | void sink(__strong A* &&); |
| 64 | |
| 65 | // CHECK-LABEL: define void @_Z5test5RU8__strongP11objc_object |
| 66 | void test5(__strong id &x) { |
| 67 | // CHECK: [[REFTMP:%.*]] = alloca {{%.*}}*, align 8 |
| 68 | // CHECK: [[I:%.*]] = alloca i32, align 4 |
| 69 | // CHECK: [[OBJ_ID:%.*]] = call i8* @llvm.objc.retain( |
| 70 | // CHECK-NEXT: [[OBJ_A:%.*]] = bitcast i8* [[OBJ_ID]] to [[A:%[a-zA-Z0-9]+]]* |
| 71 | // CHECK-NEXT: store [[A]]* [[OBJ_A]], [[A]]** [[REFTMP:%[a-zA-Z0-9]+]] |
| 72 | // CHECK-NEXT: call void @_Z4sinkOU8__strongP1A |
| 73 | sink(x); |
| 74 | // CHECK-NEXT: [[OBJ_A:%[a-zA-Z0-9]+]] = load [[A]]*, [[A]]** [[REFTMP]] |
| 75 | // CHECK-NEXT: [[OBJ_ID:%[a-zA-Z0-9]+]] = bitcast [[A]]* [[OBJ_A]] to i8* |
| 76 | // CHECK-NEXT: call void @llvm.objc.release |
| 77 | // CHECK-NEXT: [[IPTR1:%.*]] = bitcast i32* [[I]] to i8* |
| 78 | // CHECK-NEXT: call void @llvm.lifetime.start.p0i8(i64 4, i8* [[IPTR1]]) |
| 79 | // CHECK-NEXT: store i32 17, i32 |
| 80 | int i = 17; |
| 81 | // CHECK-NEXT: [[IPTR2:%.*]] = bitcast i32* [[I]] to i8* |
| 82 | // CHECK-NEXT: call void @llvm.lifetime.end.p0i8(i64 4, i8* [[IPTR2]]) |
| 83 | // CHECK-NEXT: ret void |
| 84 | } |
| 85 | |
| 86 | // CHECK-LABEL: define internal void @__cxx_global_var_init( |
| 87 | // CHECK: call i8* @_Z9getObjectv |
| 88 | // CHECK-NEXT: call i8* @llvm.objc.retainAutoreleasedReturnValue |
| 89 | const __strong id &global_ref = getObject(); |
| 90 | |
| 91 | // Note: we intentionally don't release the object. |
| 92 | |
| 93 | |