1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
|
// RUN: %clang_cc1 -triple i386-apple-darwin10 -emit-llvm -fblocks -fobjc-arc -fobjc-exceptions -fobjc-runtime=macosx-fragile-10.10 -o - %s | FileCheck %s
// RUN: %clang_cc1 -triple i386-apple-darwin10 -emit-llvm -fblocks -fobjc-arc -fobjc-exceptions -fobjc-runtime=macosx-fragile-10.10 -o - %s | FileCheck %s -check-prefix=GLOBALS
@class Opaque;
@interface Root {
Class isa;
}
@end
@interface A : Root {
Opaque *strong;
__weak Opaque *weak;
}
@end
// GLOBALS-LABEL: @OBJC_METACLASS_A
// Strong layout: scan the first word.
// GLOBALS: @OBJC_CLASS_NAME_{{.*}} = private unnamed_addr constant [2 x i8] c"\01\00"
// Weak layout: skip the first word, scan the second word.
// GLOBALS: @OBJC_CLASS_NAME_{{.*}} = private unnamed_addr constant [2 x i8] c"\11\00"
// 0x04002001
// ^ is compiled by ARC (controls interpretation of layouts)
// ^ has C++ structors (no distinction for zero-initializable)
// ^ factory (always set on non-metaclasses)
// GLOBALS: @OBJC_CLASS_A = private global {{.*}} i32 67117057
@implementation A
// CHECK-LABEL: define internal void @"\01-[A testStrong]"
// CHECK: [[SELFVAR:%.*]] = alloca ptr, align 4
- (void) testStrong {
// CHECK: [[X:%x]] = alloca ptr, align 4
// CHECK: [[SELF:%.*]] = load ptr, ptr [[SELFVAR]]
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 4
// CHECK-NEXT: [[T0:%.*]] = load ptr, ptr [[T1]]
// CHECK-NEXT: [[T2:%.*]] = call ptr @llvm.objc.retain(ptr [[T0]])
// CHECK-NEXT: store ptr [[T2]], ptr [[X]]
Opaque *x = strong;
// CHECK-NEXT: [[VALUE:%.*]] = load ptr, ptr [[X]]
// CHECK-NEXT: [[SELF:%.*]] = load ptr, ptr [[SELFVAR]]
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 4
// CHECK-NEXT: call void @llvm.objc.storeStrong(ptr [[T1]], ptr [[VALUE]])
strong = x;
// CHECK-NEXT: call void @llvm.objc.storeStrong(ptr [[X]], ptr null)
// CHECK-NEXT: ret void
}
// CHECK-LABEL: define internal void @"\01-[A testWeak]"
// CHECK: [[SELFVAR:%.*]] = alloca ptr, align 4
- (void) testWeak {
// CHECK: [[X:%x]] = alloca ptr, align 4
// CHECK: [[SELF:%.*]] = load ptr, ptr [[SELFVAR]]
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 8
// CHECK-NEXT: [[T2:%.*]] = call ptr @llvm.objc.loadWeakRetained(ptr [[T1]])
// CHECK-NEXT: store ptr [[T2]], ptr [[X]]
Opaque *x = weak;
// CHECK-NEXT: [[VALUE:%.*]] = load ptr, ptr [[X]]
// CHECK-NEXT: [[SELF:%.*]] = load ptr, ptr [[SELFVAR]]
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 8
// CHECK-NEXT: call ptr @llvm.objc.storeWeak(ptr [[T1]], ptr [[VALUE]])
weak = x;
// CHECK-NEXT: call void @llvm.objc.storeStrong(ptr [[X]], ptr null)
// CHECK-NEXT: ret void
}
// CHECK-LABEL: define internal void @"\01-[A .cxx_destruct]"
// CHECK: [[SELFVAR:%.*]] = alloca ptr, align 4
// CHECK: [[SELF:%.*]] = load ptr, ptr [[SELFVAR]]
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 8
// CHECK-NEXT: call void @llvm.objc.destroyWeak(ptr [[T1]])
// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, ptr [[SELF]], i32 4
// CHECK-NEXT: call void @llvm.objc.storeStrong(ptr [[T1]], ptr null)
// CHECK-NEXT: ret void
@end
// Test case for corner case of ivar layout.
@interface B : A {
char _b_flag;
}
@end
@interface C : B {
char _c_flag;
__unsafe_unretained id c_unsafe[5];
id c_strong[4];
__weak id c_weak[3];
id c_strong2[7];
}
@end
@implementation C @end
// Note that these layouts implicitly start at the end of the previous
// class rounded up to pointer alignment.
// GLOBALS-LABEL: @OBJC_METACLASS_C
// Strong layout: skip five, scan four, skip three, scan seven
// 'T' == 0x54, '7' == 0x37
// GLOBALS: @OBJC_CLASS_NAME_{{.*}} = private unnamed_addr constant [3 x i8] c"T7\00"
// Weak layout: skip nine, scan three
// GLOBALS: @OBJC_CLASS_NAME_{{.*}} = private unnamed_addr constant [2 x i8] c"\93\00"
extern void useBlock(void (^block)(void));
// 256 == 0x100 == starts with 1 strong
// GLOBALS: @"__block_descriptor{{.*}} = linkonce_odr hidden {{.*}}, i32 256 }
void testBlockLayoutStrong(id x) {
useBlock(^{ (void) x; });
}
// 1 == 0x001 == starts with 1 weak
// GLOBALS: @"__block_descriptor{{.*}} = linkonce_odr hidden {{.*}}, i32 1 }
void testBlockLayoutWeak(__weak id x) {
useBlock(^{ (void) x; });
}
// CHECK-LABEL: define{{.*}} void @testCatch()
// CHECK: [[X:%.*]] = alloca ptr, align 4
// CHECK: [[Y:%.*]] = alloca ptr, align 4
// CHECK: call void @objc_exception_try_enter
// CHECK: br i1
// CHECK: call void @checkpoint(i32 noundef 0)
// CHECK: call void @objc_exception_try_exit
// CHECK: br label
// CHECK: call void @checkpoint(i32 noundef 3)
// CHECK: [[EXN:%.*]] = call ptr @objc_exception_extract
// CHECK: call i32 @objc_exception_match(
// CHECK: br i1
// CHECK: [[T2:%.*]] = call ptr @llvm.objc.retain(ptr [[EXN]])
// CHECK: store ptr [[T2]], ptr [[X]]
// CHECK: call void @checkpoint(i32 noundef 1)
// CHECK: call void @llvm.objc.storeStrong(ptr [[X]], ptr null)
// CHECK: br label
// CHECK: [[T0:%.*]] = call ptr @llvm.objc.retain(ptr [[EXN]])
// CHECK: store ptr [[T0]], ptr [[Y]]
// CHECK: call void @checkpoint(i32 noundef 2)
// CHECK: call void @llvm.objc.storeStrong(ptr [[Y]], ptr null)
extern void checkpoint(int n);
void testCatch(void) {
@try {
checkpoint(0);
} @catch (A *x) {
checkpoint(1);
} @catch (id y) {
checkpoint(2);
}
checkpoint(3);
}
|