134 lines
3.7 KiB
Mathematica
134 lines
3.7 KiB
Mathematica
|
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -fobjc-arc -fobjc-runtime-has-weak -o - %s | FileCheck %s
|
||
|
|
||
|
// CHECK: %[[STRUCT_TRIVIAL:.*]] = type { i32 }
|
||
|
// CHECK: %[[STRUCT_TRIVIALBIG:.*]] = type { [64 x i32] }
|
||
|
// CHECK: %[[STRUCT_STRONG:.*]] = type { i8* }
|
||
|
// CHECK: %[[STRUCT_WEAK:.*]] = type { i8* }
|
||
|
|
||
|
typedef struct {
|
||
|
int x;
|
||
|
} Trivial;
|
||
|
|
||
|
typedef struct {
|
||
|
int x[64];
|
||
|
} TrivialBig;
|
||
|
|
||
|
typedef struct {
|
||
|
id x;
|
||
|
} Strong;
|
||
|
|
||
|
typedef struct {
|
||
|
__weak id x;
|
||
|
} Weak;
|
||
|
|
||
|
// CHECK: define{{.*}} i32 @testTrivial()
|
||
|
// CHECK: %[[RETVAL:.*]] = alloca %[[STRUCT_TRIVIAL]], align 4
|
||
|
// CHECK-NEXT: call void @func0(%[[STRUCT_TRIVIAL]]* %[[RETVAL]])
|
||
|
// CHECK-NOT: memcpy
|
||
|
// CHECK: ret i32 %
|
||
|
|
||
|
void func0(Trivial *);
|
||
|
|
||
|
Trivial testTrivial(void) {
|
||
|
Trivial a;
|
||
|
func0(&a);
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
void func1(TrivialBig *);
|
||
|
|
||
|
// CHECK: define{{.*}} void @testTrivialBig(%[[STRUCT_TRIVIALBIG]]* noalias sret(%[[STRUCT_TRIVIALBIG]]) align 4 %[[AGG_RESULT:.*]])
|
||
|
// CHECK: call void @func1(%[[STRUCT_TRIVIALBIG]]* %[[AGG_RESULT]])
|
||
|
// CHECK-NEXT: ret void
|
||
|
|
||
|
TrivialBig testTrivialBig(void) {
|
||
|
TrivialBig a;
|
||
|
func1(&a);
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
// CHECK: define{{.*}} i8* @testStrong()
|
||
|
// CHECK: %[[RETVAL:.*]] = alloca %[[STRUCT_STRONG]], align 8
|
||
|
// CHECK: %[[NRVO:.*]] = alloca i1, align 1
|
||
|
// CHECK: %[[V0:.*]] = bitcast %[[STRUCT_STRONG]]* %[[RETVAL]] to i8**
|
||
|
// CHECK: call void @__default_constructor_8_s0(i8** %[[V0]])
|
||
|
// CHECK: store i1 true, i1* %[[NRVO]], align 1
|
||
|
// CHECK: %[[NRVO_VAL:.*]] = load i1, i1* %[[NRVO]], align 1
|
||
|
// CHECK: br i1 %[[NRVO_VAL]],
|
||
|
|
||
|
// CHECK: %[[V1:.*]] = bitcast %[[STRUCT_STRONG]]* %[[RETVAL]] to i8**
|
||
|
// CHECK: call void @__destructor_8_s0(i8** %[[V1]])
|
||
|
// CHECK: br
|
||
|
|
||
|
// CHECK: %[[COERCE_DIVE:.*]] = getelementptr inbounds %[[STRUCT_STRONG]], %[[STRUCT_STRONG]]* %[[RETVAL]], i32 0, i32 0
|
||
|
// CHECK: %[[V2:.*]] = load i8*, i8** %[[COERCE_DIVE]], align 8
|
||
|
// CHECK: ret i8* %[[V2]]
|
||
|
|
||
|
Strong testStrong(void) {
|
||
|
Strong a;
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
// CHECK: define{{.*}} void @testWeak(%[[STRUCT_WEAK]]* noalias sret(%[[STRUCT_WEAK]]) align 8 %[[AGG_RESULT:.*]])
|
||
|
// CHECK: %[[NRVO:.*]] = alloca i1, align 1
|
||
|
// CHECK: %[[V0:.*]] = bitcast %[[STRUCT_WEAK]]* %[[AGG_RESULT]] to i8**
|
||
|
// CHECK: call void @__default_constructor_8_w0(i8** %[[V0]])
|
||
|
// CHECK: store i1 true, i1* %[[NRVO]], align 1
|
||
|
// CHECK: %[[NRVO_VAL:.*]] = load i1, i1* %[[NRVO]], align 1
|
||
|
// CHECK: br i1 %[[NRVO_VAL]],
|
||
|
|
||
|
// CHECK: %[[V1:.*]] = bitcast %[[STRUCT_WEAK]]* %[[AGG_RESULT]] to i8**
|
||
|
// CHECK: call void @__destructor_8_w0(i8** %[[V1]])
|
||
|
// CHECK: br
|
||
|
|
||
|
// CHECK-NOT: call
|
||
|
// CHECK: ret void
|
||
|
|
||
|
Weak testWeak(void) {
|
||
|
Weak a;
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
// CHECK: define{{.*}} void @testWeak2(
|
||
|
// CHECK: call void @__default_constructor_8_w0(
|
||
|
// CHECK: call void @__default_constructor_8_w0(
|
||
|
// CHECK: call void @__copy_constructor_8_8_w0(
|
||
|
// CHECK: call void @__copy_constructor_8_8_w0(
|
||
|
// CHECK: call void @__destructor_8_w0(
|
||
|
// CHECK: call void @__destructor_8_w0(
|
||
|
|
||
|
Weak testWeak2(int c) {
|
||
|
Weak a, b;
|
||
|
if (c)
|
||
|
return a;
|
||
|
else
|
||
|
return b;
|
||
|
}
|
||
|
|
||
|
// CHECK: define internal void @"\01-[C1 foo1]"(%[[STRUCT_WEAK]]* noalias sret(%[[STRUCT_WEAK]]) align 8 %[[AGG_RESULT:.*]], %{{.*}}* %{{.*}}, i8* %{{.*}})
|
||
|
// CHECK: %[[NRVO:.*]] = alloca i1, align 1
|
||
|
// CHECK: %[[V0:.*]] = bitcast %[[STRUCT_WEAK]]* %[[AGG_RESULT]] to i8**
|
||
|
// CHECK: call void @__default_constructor_8_w0(i8** %[[V0]])
|
||
|
// CHECK: store i1 true, i1* %[[NRVO]], align 1
|
||
|
// CHECK: %[[NRVO_VAL:.*]] = load i1, i1* %[[NRVO]], align 1
|
||
|
// CHECK: br i1 %[[NRVO_VAL]],
|
||
|
|
||
|
// CHECK: %[[V1:.*]] = bitcast %[[STRUCT_WEAK]]* %[[AGG_RESULT]] to i8**
|
||
|
// CHECK: call void @__destructor_8_w0(i8** %[[V1]])
|
||
|
// CHECK: br
|
||
|
|
||
|
// CHECK-NOT: call
|
||
|
// CHECK: ret void
|
||
|
|
||
|
__attribute__((objc_root_class))
|
||
|
@interface C1
|
||
|
- (Weak)foo1;
|
||
|
@end
|
||
|
|
||
|
@implementation C1
|
||
|
- (Weak)foo1 {
|
||
|
Weak a;
|
||
|
return a;
|
||
|
}
|
||
|
@end
|