2015-11-19 10:28:03 +08:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -fobjc-runtime=macosx-10.10 -emit-llvm -fblocks -fobjc-weak -o - %s | FileCheck %s -check-prefix=CHECK -check-prefix=CHECK-MODERN
|
|
|
|
// RUN: %clang_cc1 -triple i386-apple-darwin10 -fobjc-runtime=macosx-fragile-10.10 -emit-llvm -fblocks -fobjc-weak -o - %s | FileCheck %s -check-prefix=CHECK -check-prefix=CHECK-FRAGILE
|
|
|
|
|
|
|
|
@interface Object
|
|
|
|
- (instancetype) retain;
|
|
|
|
- (void) run;
|
|
|
|
@end
|
|
|
|
|
|
|
|
// CHECK-MODERN: @OBJC_CLASS_NAME_{{.*}} = {{.*}} c"\01\00"
|
|
|
|
// CHECK-MODERN: @"\01l_OBJC_CLASS_RO_$_Foo" = {{.*}} { i32 772
|
|
|
|
// 772 == 0x304
|
|
|
|
// ^ HasMRCWeakIvars
|
|
|
|
// ^ HasCXXDestructorOnly
|
|
|
|
// ^ HasCXXStructors
|
|
|
|
|
|
|
|
// CHECK-FRAGILE: @OBJC_CLASS_NAME_{{.*}} = {{.*}} c"\01\00"
|
|
|
|
// CHECK-FRAGILE: @OBJC_CLASS_Foo = {{.*}} i32 134225921,
|
|
|
|
// 134225921 == 0x08002001
|
|
|
|
// ^ HasMRCWeakIvars
|
|
|
|
// ^ HasCXXStructors
|
|
|
|
// ^ Factory
|
|
|
|
@interface Foo : Object {
|
|
|
|
__weak id ivar;
|
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation Foo
|
|
|
|
// CHECK-LABEL: define internal void @"\01-[Foo .cxx_destruct]"
|
|
|
|
// CHECK: call void @objc_destroyWeak
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
void test1(__weak id x) {}
|
|
|
|
// CHECK-LABEL: define void @_Z5test1P11objc_object(
|
|
|
|
// CHECK: [[X:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: objc_initWeak
|
|
|
|
// CHECK-NEXT: objc_destroyWeak
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
void test2(id y) {
|
|
|
|
__weak id z = y;
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test2P11objc_object(
|
|
|
|
// CHECK: [[Y:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: [[Z:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: store
|
|
|
|
// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[Y]]
|
|
|
|
// CHECK-NEXT: call i8* @objc_initWeak(i8** [[Z]], i8* [[T0]])
|
|
|
|
// CHECK-NEXT: call void @objc_destroyWeak(i8** [[Z]])
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
void test3(id y) {
|
|
|
|
__weak id z;
|
|
|
|
z = y;
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test3P11objc_object(
|
|
|
|
// CHECK: [[Y:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: [[Z:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: store
|
|
|
|
// CHECK-NEXT: store i8* null, i8** [[Z]]
|
|
|
|
// CHECK-NEXT: [[T0:%.*]] = load i8*, i8** [[Y]]
|
|
|
|
// CHECK-NEXT: call i8* @objc_storeWeak(i8** [[Z]], i8* [[T0]])
|
|
|
|
// CHECK-NEXT: call void @objc_destroyWeak(i8** [[Z]])
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
void test4(__weak id *p) {
|
|
|
|
id y = *p;
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test4PU6__weakP11objc_object(
|
|
|
|
// CHECK: [[P:%.*]] = alloca i8**,
|
|
|
|
// CHECK-NEXT: [[Y:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: store
|
|
|
|
// CHECK-NEXT: [[T0:%.*]] = load i8**, i8*** [[P]]
|
|
|
|
// CHECK-NEXT: [[T1:%.*]] = call i8* @objc_loadWeak(i8** [[T0]])
|
|
|
|
// CHECK-NEXT: store i8* [[T1]], i8** [[Y]]
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
void test5(__weak id *p) {
|
|
|
|
id y = [*p retain];
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test5PU6__weakP11objc_object
|
|
|
|
// CHECK: [[P:%.*]] = alloca i8**,
|
|
|
|
// CHECK-NEXT: [[Y:%.*]] = alloca i8*,
|
|
|
|
// CHECK-NEXT: store
|
|
|
|
// CHECK-NEXT: [[T0:%.*]] = load i8**, i8*** [[P]]
|
|
|
|
// CHECK-NEXT: [[T1:%.*]] = call i8* @objc_loadWeakRetained(i8** [[T0]])
|
|
|
|
// CHECK-NEXT: store i8* [[T1]], i8** [[Y]]
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
void test6(__weak Foo **p) {
|
|
|
|
Foo *y = [*p retain];
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test6PU6__weakP3Foo
|
|
|
|
// CHECK: [[P:%.*]] = alloca [[FOO:%.*]]**,
|
|
|
|
// CHECK-NEXT: [[Y:%.*]] = alloca [[FOO]]*,
|
|
|
|
// CHECK-NEXT: store
|
|
|
|
// CHECK-NEXT: [[T0:%.*]] = load [[FOO]]**, [[FOO]]*** [[P]]
|
|
|
|
// CHECK-NEXT: [[T1:%.*]] = bitcast [[FOO]]** [[T0]] to i8**
|
|
|
|
// CHECK-NEXT: [[T2:%.*]] = call i8* @objc_loadWeakRetained(i8** [[T1]])
|
|
|
|
// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to [[FOO]]*
|
|
|
|
// CHECK-NEXT: store [[FOO]]* [[T3]], [[FOO]]** [[Y]]
|
|
|
|
// CHECK-NEXT: ret void
|
|
|
|
|
|
|
|
extern "C" id get_object(void);
|
|
|
|
extern "C" void use_block(void (^)(void));
|
|
|
|
|
|
|
|
void test7(void) {
|
|
|
|
__weak Foo *p = get_object();
|
|
|
|
use_block(^{ [p run ]; });
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test7v
|
|
|
|
// CHECK: [[P:%.*]] = alloca [[FOO]]*,
|
|
|
|
// CHECK: [[T0:%.*]] = call i8* @get_object()
|
|
|
|
// CHECK-NEXT: [[T1:%.*]] = bitcast i8* [[T0]] to [[FOO]]*
|
|
|
|
// CHECK-NEXT: [[T2:%.*]] = bitcast [[FOO]]** [[P]] to i8**
|
|
|
|
// CHECK-NEXT: [[T3:%.*]] = bitcast [[FOO]]* [[T1]] to i8*
|
|
|
|
// CHECK-NEXT: call i8* @objc_initWeak(i8** [[T2]], i8* [[T3]])
|
|
|
|
// CHECK: call void @objc_copyWeak
|
|
|
|
// CHECK: call void @use_block
|
|
|
|
// CHECK: call void @objc_destroyWeak
|
|
|
|
|
2018-08-10 23:09:24 +08:00
|
|
|
// CHECK-LABEL: define linkonce_odr hidden void @__copy_helper_block
|
2015-11-19 10:28:03 +08:00
|
|
|
// CHECK: @objc_copyWeak
|
|
|
|
|
2018-08-10 23:09:24 +08:00
|
|
|
// CHECK-LABEL: define linkonce_odr hidden void @__destroy_helper_block
|
2015-11-19 10:28:03 +08:00
|
|
|
// CHECK: @objc_destroyWeak
|
|
|
|
|
|
|
|
void test8(void) {
|
|
|
|
__block __weak Foo *p = get_object();
|
|
|
|
use_block(^{ [p run ]; });
|
|
|
|
}
|
|
|
|
// CHECK-LABEL: define void @_Z5test8v
|
|
|
|
// CHECK: call i8* @objc_initWeak
|
|
|
|
// CHECK-NOT: call void @objc_copyWeak
|
|
|
|
// CHECK: call void @use_block
|
|
|
|
// CHECK: call void @objc_destroyWeak
|
|
|
|
|
|
|
|
// CHECK-LABEL: define internal void @__Block_byref_object_copy
|
|
|
|
// CHECK: call void @objc_moveWeak
|
|
|
|
|
|
|
|
// CHECK-LABEL: define internal void @__Block_byref_object_dispose
|
|
|
|
// CHECK: call void @objc_destroyWeak
|
|
|
|
|
|
|
|
// CHECK-LABEL: define void @_Z14test9_baselinev()
|
2018-08-10 23:09:24 +08:00
|
|
|
// CHECK: define linkonce_odr hidden void @__copy_helper
|
|
|
|
// CHECK: define linkonce_odr hidden void @__destroy_helper
|
2015-11-19 10:28:03 +08:00
|
|
|
void test9_baseline(void) {
|
|
|
|
Foo *p = get_object();
|
|
|
|
use_block(^{ [p run]; });
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK-LABEL: define void @_Z5test9v()
|
|
|
|
// CHECK-NOT: define internal void @__copy_helper
|
|
|
|
// CHECK-NOT: define internal void @__destroy_helper
|
|
|
|
// CHECK: define void @_Z9test9_finv()
|
|
|
|
void test9(void) {
|
|
|
|
__unsafe_unretained Foo *p = get_object();
|
|
|
|
use_block(^{ [p run]; });
|
|
|
|
}
|
|
|
|
void test9_fin() {}
|
|
|
|
|
|
|
|
// CHECK-LABEL: define void @_Z6test10v()
|
|
|
|
// CHECK-NOT: define internal void @__copy_helper
|
|
|
|
// CHECK-NOT: define internal void @__destroy_helper
|
|
|
|
// CHECK: define void @_Z10test10_finv()
|
|
|
|
void test10(void) {
|
|
|
|
typedef __unsafe_unretained Foo *UnsafeFooPtr;
|
|
|
|
UnsafeFooPtr p = get_object();
|
|
|
|
use_block(^{ [p run]; });
|
|
|
|
}
|
|
|
|
void test10_fin() {}
|
|
|
|
|
|
|
|
// CHECK-LABEL: define weak_odr void @_Z6test11ILj0EEvv()
|
|
|
|
// CHECK-NOT: define internal void @__copy_helper
|
|
|
|
// CHECK-NOT: define internal void @__destroy_helper
|
|
|
|
// CHECK: define void @_Z10test11_finv()
|
|
|
|
template <unsigned i> void test11(void) {
|
|
|
|
typedef __unsafe_unretained Foo *UnsafeFooPtr;
|
|
|
|
UnsafeFooPtr p = get_object();
|
|
|
|
use_block(^{ [p run]; });
|
|
|
|
}
|
|
|
|
template void test11<0>();
|
|
|
|
void test11_fin() {}
|