xref: /aosp_15_r20/external/clang/test/CodeGenObjC/arc-property.m (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -fobjc-arc -emit-llvm %s -o - | FileCheck %s
2*67e74705SXin Li
3*67e74705SXin Li// rdar://problem/10290317
4*67e74705SXin Li@interface Test0
5*67e74705SXin Li- (void) setValue: (id) x;
6*67e74705SXin Li@end
7*67e74705SXin Livoid test0(Test0 *t0, id value) {
8*67e74705SXin Li  t0.value = value;
9*67e74705SXin Li}
10*67e74705SXin Li// CHECK-LABEL: define void @test0(
11*67e74705SXin Li// CHECK: call void @objc_storeStrong
12*67e74705SXin Li// CHECK: call void @objc_storeStrong
13*67e74705SXin Li// CHECK: @objc_msgSend
14*67e74705SXin Li// CHECK: call void @objc_storeStrong(
15*67e74705SXin Li// CHECK: call void @objc_storeStrong(
16*67e74705SXin Li
17*67e74705SXin Listruct S1 { Class isa; };
18*67e74705SXin Li@interface Test1
19*67e74705SXin Li@property (nonatomic, strong) __attribute__((NSObject)) struct S1 *pointer;
20*67e74705SXin Li@end
21*67e74705SXin Li@implementation Test1
22*67e74705SXin Li@synthesize pointer;
23*67e74705SXin Li@end
24*67e74705SXin Li//   The getter should be a simple load.
25*67e74705SXin Li// CHECK:    define internal [[S1:%.*]]* @"\01-[Test1 pointer]"(
26*67e74705SXin Li// CHECK:      [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test1.pointer"
27*67e74705SXin Li// CHECK-NEXT: [[T0:%.*]] = bitcast [[TEST1:%.*]]* {{%.*}} to i8*
28*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = getelementptr inbounds i8, i8* [[T0]], i64 [[OFFSET]]
29*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = bitcast i8* [[T1]] to [[S1]]**
30*67e74705SXin Li// CHECK-NEXT: [[T3:%.*]] = load [[S1]]*, [[S1]]** [[T2]], align 8
31*67e74705SXin Li// CHECK-NEXT: ret [[S1]]* [[T3]]
32*67e74705SXin Li
33*67e74705SXin Li//   The setter should be using objc_setProperty.
34*67e74705SXin Li// CHECK:    define internal void @"\01-[Test1 setPointer:]"(
35*67e74705SXin Li// CHECK:      [[T0:%.*]] = bitcast [[TEST1]]* {{%.*}} to i8*
36*67e74705SXin Li// CHECK-NEXT: [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test1.pointer"
37*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = load [[S1]]*, [[S1]]** {{%.*}}
38*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = bitcast [[S1]]* [[T1]] to i8*
39*67e74705SXin Li// CHECK-NEXT: call void @objc_setProperty(i8* [[T0]], i8* {{%.*}}, i64 [[OFFSET]], i8* [[T2]], i1 zeroext false, i1 zeroext false)
40*67e74705SXin Li// CHECK-NEXT: ret void
41*67e74705SXin Li
42*67e74705SXin Li
43*67e74705SXin Li// rdar://problem/12039404
44*67e74705SXin Li@interface Test2 {
45*67e74705SXin Li@private
46*67e74705SXin Li  Class _theClass;
47*67e74705SXin Li}
48*67e74705SXin Li@property (copy) Class theClass;
49*67e74705SXin Li@end
50*67e74705SXin Li
51*67e74705SXin Listatic Class theGlobalClass;
52*67e74705SXin Li@implementation Test2
53*67e74705SXin Li@synthesize theClass = _theClass;
54*67e74705SXin Li- (void) test {
55*67e74705SXin Li  _theClass = theGlobalClass;
56*67e74705SXin Li}
57*67e74705SXin Li@end
58*67e74705SXin Li// CHECK:    define internal void @"\01-[Test2 test]"(
59*67e74705SXin Li// CHECK:      [[T0:%.*]] = load i8*, i8** @theGlobalClass, align 8
60*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = load [[TEST2:%.*]]*, [[TEST2:%.*]]**
61*67e74705SXin Li// CHECK-NEXT: [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test2._theClass"
62*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = bitcast [[TEST2]]* [[T1]] to i8*
63*67e74705SXin Li// CHECK-NEXT: [[T3:%.*]] = getelementptr inbounds i8, i8* [[T2]], i64 [[OFFSET]]
64*67e74705SXin Li// CHECK-NEXT: [[T4:%.*]] = bitcast i8* [[T3]] to i8**
65*67e74705SXin Li// CHECK-NEXT: call void @objc_storeStrong(i8** [[T4]], i8* [[T0]]) [[NUW:#[0-9]+]]
66*67e74705SXin Li// CHECK-NEXT: ret void
67*67e74705SXin Li
68*67e74705SXin Li// CHECK:    define internal i8* @"\01-[Test2 theClass]"(
69*67e74705SXin Li// CHECK:      [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test2._theClass"
70*67e74705SXin Li// CHECK-NEXT: [[T0:%.*]] = tail call i8* @objc_getProperty(i8* {{.*}}, i8* {{.*}}, i64 [[OFFSET]], i1 zeroext true)
71*67e74705SXin Li// CHECK-NEXT: ret i8* [[T0]]
72*67e74705SXin Li
73*67e74705SXin Li// CHECK:    define internal void @"\01-[Test2 setTheClass:]"(
74*67e74705SXin Li// CHECK:      [[T0:%.*]] = bitcast [[TEST2]]* {{%.*}} to i8*
75*67e74705SXin Li// CHECK-NEXT: [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test2._theClass"
76*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = load i8*, i8** {{%.*}}
77*67e74705SXin Li// CHECK-NEXT: call void @objc_setProperty(i8* [[T0]], i8* {{%.*}}, i64 [[OFFSET]], i8* [[T1]], i1 zeroext true, i1 zeroext true)
78*67e74705SXin Li// CHECK-NEXT: ret void
79*67e74705SXin Li
80*67e74705SXin Li// CHECK:    define internal void @"\01-[Test2 .cxx_destruct]"(
81*67e74705SXin Li// CHECK:      [[T0:%.*]] = load [[TEST2]]*, [[TEST2]]**
82*67e74705SXin Li// CHECK-NEXT: [[OFFSET:%.*]] = load i64, i64* @"OBJC_IVAR_$_Test2._theClass"
83*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = bitcast [[TEST2]]* [[T0]] to i8*
84*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = getelementptr inbounds i8, i8* [[T1]], i64 [[OFFSET]]
85*67e74705SXin Li// CHECK-NEXT: [[T3:%.*]] = bitcast i8* [[T2]] to i8**
86*67e74705SXin Li// CHECK-NEXT: call void @objc_storeStrong(i8** [[T3]], i8* null) [[NUW]]
87*67e74705SXin Li// CHECK-NEXT: ret void
88*67e74705SXin Li
89*67e74705SXin Li// rdar://13115896
90*67e74705SXin Li@interface Test3
91*67e74705SXin Li@property id copyMachine;
92*67e74705SXin Li@end
93*67e74705SXin Li
94*67e74705SXin Livoid test3(Test3 *t) {
95*67e74705SXin Li  id x = t.copyMachine;
96*67e74705SXin Li  x = [t copyMachine];
97*67e74705SXin Li}
98*67e74705SXin Li// CHECK:    define void @test3([[TEST3:%.*]]*
99*67e74705SXin Li//   Prologue.
100*67e74705SXin Li// CHECK:      [[T:%.*]] = alloca [[TEST3]]*,
101*67e74705SXin Li// CHECK-NEXT: [[X:%.*]] = alloca i8*,
102*67e74705SXin Li//   Property access.
103*67e74705SXin Li// CHECK:      [[T0:%.*]] = load [[TEST3]]*, [[TEST3]]** [[T]],
104*67e74705SXin Li// CHECK-NEXT: [[SEL:%.*]] = load i8*, i8** @OBJC_SELECTOR_REFERENCES
105*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = bitcast [[TEST3]]* [[T0]] to i8*
106*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = call i8* bitcast ({{.*}} @objc_msgSend to {{.*}})(i8* [[T1]], i8* [[SEL]])
107*67e74705SXin Li// CHECK-NEXT: store i8* [[T2]], i8** [[X]],
108*67e74705SXin Li//   Message send.
109*67e74705SXin Li// CHECK-NEXT: [[T0:%.*]] = load [[TEST3]]*, [[TEST3]]** [[T]],
110*67e74705SXin Li// CHECK-NEXT: [[SEL:%.*]] = load i8*, i8** @OBJC_SELECTOR_REFERENCES
111*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = bitcast [[TEST3]]* [[T0]] to i8*
112*67e74705SXin Li// CHECK-NEXT: [[T2:%.*]] = call i8* bitcast ({{.*}} @objc_msgSend to {{.*}})(i8* [[T1]], i8* [[SEL]])
113*67e74705SXin Li// CHECK-NEXT: [[T3:%.*]] = load i8*, i8** [[X]],
114*67e74705SXin Li// CHECK-NEXT: store i8* [[T2]], i8** [[X]],
115*67e74705SXin Li// CHECK-NEXT: call void @objc_release(i8* [[T3]])
116*67e74705SXin Li//   Epilogue.
117*67e74705SXin Li// CHECK-NEXT: call void @objc_storeStrong(i8** [[X]], i8* null)
118*67e74705SXin Li// CHECK-NEXT: [[T0:%.*]] = bitcast [[TEST3]]** [[T]] to i8**
119*67e74705SXin Li// CHECK-NEXT: call void @objc_storeStrong(i8** [[T0]], i8* null)
120*67e74705SXin Li// CHECK-NEXT: ret void
121*67e74705SXin Li
122*67e74705SXin Li@implementation Test3
123*67e74705SXin Li- (id) copyMachine {
124*67e74705SXin Li  extern id test3_helper(void);
125*67e74705SXin Li  return test3_helper();
126*67e74705SXin Li}
127*67e74705SXin Li// CHECK:    define internal i8* @"\01-[Test3 copyMachine]"(
128*67e74705SXin Li// CHECK:      [[T0:%.*]] = call i8* @test3_helper()
129*67e74705SXin Li// CHECK-NEXT: [[T1:%.*]] = call i8* @objc_retainAutoreleasedReturnValue(i8* [[T0]])
130*67e74705SXin Li// CHECK-NEXT: ret i8* [[T1]]
131*67e74705SXin Li- (void) setCopyMachine: (id) x {}
132*67e74705SXin Li@end
133*67e74705SXin Li
134*67e74705SXin Li// CHECK: attributes [[NUW]] = { nounwind }
135