|  | // 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 %[[AGG_RESULT:.*]]) | 
|  | // CHECK-NOT: alloca | 
|  | // 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 %[[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 %[[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 |