// RUN: %clang_cc1 -fobjc-arc -fobjc-runtime-has-weak -fblocks -triple x86_64-apple-darwin10.0.0 -emit-llvm -o - %s | FileCheck %s -check-prefix=CHECK -check-prefix=UNOPT // RUN: %clang_cc1 -fobjc-arc -fobjc-runtime-has-weak -fblocks -triple x86_64-apple-darwin10.0.0 -emit-llvm -o - %s -O -disable-llvm-passes | FileCheck %s -check-prefix=CHECK -check-prefix=OPT typedef __strong id strong_id; typedef __weak id weak_id; // CHECK-LABEL: define{{.*}} void @_Z8test_newP11objc_object void test_new(id invalue) { // CHECK: [[INVALUEADDR:%.*]] = alloca ptr // UNOPT-NEXT: store ptr null, ptr [[INVALUEADDR]] // UNOPT-NEXT: call void @llvm.objc.storeStrong(ptr [[INVALUEADDR]], ptr [[INVALUE:%.*]]) // OPT-NEXT: [[T0:%.*]] = call ptr @llvm.objc.retain(ptr [[INVALUE:%.*]]) // OPT-NEXT: store ptr [[T0]], ptr [[INVALUEADDR]] // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // CHECK-NEXT: store ptr null, ptr new strong_id; // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // UNOPT-NEXT: store ptr null, ptr // OPT-NEXT: call ptr @llvm.objc.initWeak(ptr {{.*}}, ptr null) new weak_id; // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // CHECK-NEXT: store ptr null, ptr new __strong id; // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // UNOPT-NEXT: store ptr null, ptr // OPT-NEXT: call ptr @llvm.objc.initWeak(ptr {{.*}}, ptr null) new __weak id; // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // CHECK: call ptr @llvm.objc.retain // CHECK: store ptr new __strong id(invalue); // CHECK: [[CALL:%.*]] = call noalias noundef nonnull ptr @_Znwm // CHECK: call ptr @llvm.objc.initWeak new __weak id(invalue); // UNOPT: call void @llvm.objc.storeStrong // OPT: call void @llvm.objc.release // CHECK: ret void } // CHECK-LABEL: define{{.*}} void @_Z14test_array_new void test_array_new() { // CHECK: call noalias noundef nonnull ptr @_Znam // CHECK: store i64 17, ptr // CHECK: call void @llvm.memset.p0.i64 new strong_id[17]; // CHECK: call noalias noundef nonnull ptr @_Znam // CHECK: store i64 17, ptr // CHECK: call void @llvm.memset.p0.i64 new weak_id[17]; // CHECK: ret void } // CHECK-LABEL: define{{.*}} void @_Z11test_deletePU8__strongP11objc_objectPU6__weakS0_ void test_delete(__strong id *sptr, __weak id *wptr) { // CHECK: br i1 // UNOPT: call void @llvm.objc.storeStrong(ptr {{.*}}, ptr null) // OPT: load ptr, ptr // OPT-NEXT: call void @llvm.objc.release // CHECK: call void @_ZdlPv delete sptr; // CHECK: call void @llvm.objc.destroyWeak // CHECK: call void @_ZdlPv delete wptr; // CHECK: ret void } // CHECK-LABEL: define{{.*}} void @_Z17test_array_deletePU8__strongP11objc_objectPU6__weakS0_ void test_array_delete(__strong id *sptr, __weak id *wptr) { // CHECK: icmp eq ptr [[BEGIN:%.*]], null // CHECK: [[LEN:%.*]] = load i64, ptr {{%.*}} // CHECK: [[END:%.*]] = getelementptr inbounds ptr, ptr [[BEGIN]], i64 [[LEN]] // CHECK-NEXT: icmp eq ptr [[BEGIN]], [[END]] // CHECK: [[PAST:%.*]] = phi ptr [ [[END]], {{%.*}} ], [ [[CUR:%.*]], // CHECK-NEXT: [[CUR]] = getelementptr inbounds ptr, ptr [[PAST]], i64 -1 // UNOPT-NEXT: call void @llvm.objc.storeStrong(ptr [[CUR]], ptr null) // OPT-NEXT: [[T0:%.*]] = load ptr, ptr [[CUR]] // OPT-NEXT: llvm.objc.release(ptr [[T0]]) // CHECK-NEXT: icmp eq ptr [[CUR]], [[BEGIN]] // CHECK: call void @_ZdaPv delete [] sptr; // CHECK: icmp eq ptr [[BEGIN:%.*]], null // CHECK: [[LEN:%.*]] = load i64, ptr {{%.*}} // CHECK: [[END:%.*]] = getelementptr inbounds ptr, ptr [[BEGIN]], i64 [[LEN]] // CHECK-NEXT: icmp eq ptr [[BEGIN]], [[END]] // CHECK: [[PAST:%.*]] = phi ptr [ [[END]], {{%.*}} ], [ [[CUR:%.*]], // CHECK-NEXT: [[CUR]] = getelementptr inbounds ptr, ptr [[PAST]], i64 -1 // CHECK-NEXT: call void @llvm.objc.destroyWeak(ptr [[CUR]]) // CHECK-NEXT: icmp eq ptr [[CUR]], [[BEGIN]] // CHECK: call void @_ZdaPv delete [] wptr; }