aboutsummaryrefslogtreecommitdiff
path: root/test/CodeGenObjC
diff options
context:
space:
mode:
Diffstat (limited to 'test/CodeGenObjC')
-rw-r--r--test/CodeGenObjC/Inputs/nsvalue-boxed-expressions-support.h63
-rw-r--r--test/CodeGenObjC/nsvalue-objc-boxable-ios-arc.m126
-rw-r--r--test/CodeGenObjC/nsvalue-objc-boxable-ios.m114
-rw-r--r--test/CodeGenObjC/nsvalue-objc-boxable-mac-arc.m130
-rw-r--r--test/CodeGenObjC/nsvalue-objc-boxable-mac.m118
5 files changed, 551 insertions, 0 deletions
diff --git a/test/CodeGenObjC/Inputs/nsvalue-boxed-expressions-support.h b/test/CodeGenObjC/Inputs/nsvalue-boxed-expressions-support.h
new file mode 100644
index 000000000000..cbcf882b27e3
--- /dev/null
+++ b/test/CodeGenObjC/Inputs/nsvalue-boxed-expressions-support.h
@@ -0,0 +1,63 @@
+#ifndef NSVALUE_BOXED_EXPRESSIONS_SUPPORT_H
+#define NSVALUE_BOXED_EXPRESSIONS_SUPPORT_H
+
+#define BOXABLE __attribute__((objc_boxable))
+
+typedef unsigned long NSUInteger;
+typedef double CGFloat;
+
+typedef struct BOXABLE _NSRange {
+ NSUInteger location;
+ NSUInteger length;
+} NSRange;
+
+typedef struct BOXABLE _NSPoint {
+ CGFloat x;
+ CGFloat y;
+} NSPoint;
+
+typedef struct BOXABLE _NSSize {
+ CGFloat width;
+ CGFloat height;
+} NSSize;
+
+typedef struct BOXABLE _NSRect {
+ NSPoint origin;
+ NSSize size;
+} NSRect;
+
+struct CGPoint {
+ CGFloat x;
+ CGFloat y;
+};
+typedef struct BOXABLE CGPoint CGPoint;
+
+struct CGSize {
+ CGFloat width;
+ CGFloat height;
+};
+typedef struct BOXABLE CGSize CGSize;
+
+struct CGRect {
+ CGPoint origin;
+ CGSize size;
+};
+typedef struct BOXABLE CGRect CGRect;
+
+struct NSEdgeInsets {
+ CGFloat top;
+ CGFloat left;
+ CGFloat bottom;
+ CGFloat right;
+};
+typedef struct BOXABLE NSEdgeInsets NSEdgeInsets;
+
+@interface NSValue
+
++ (NSValue *)valueWithBytes:(const void *)value objCType:(const char *)type;
+
+@end
+
+NSRange getRange();
+
+#endif // NSVALUE_BOXED_EXPRESSIONS_SUPPORT_H
diff --git a/test/CodeGenObjC/nsvalue-objc-boxable-ios-arc.m b/test/CodeGenObjC/nsvalue-objc-boxable-ios-arc.m
new file mode 100644
index 000000000000..75e04dbd0272
--- /dev/null
+++ b/test/CodeGenObjC/nsvalue-objc-boxable-ios-arc.m
@@ -0,0 +1,126 @@
+// RUN: %clang_cc1 -I %S/Inputs -triple armv7-apple-ios8.0.0 -emit-llvm -fobjc-arc -O2 -disable-llvm-optzns -o - %s | FileCheck %s
+
+#import "nsvalue-boxed-expressions-support.h"
+
+// CHECK: [[CLASS:@.*]] = external global %struct._class_t
+// CHECK: [[NSVALUE:@.*]] = {{.*}}[[CLASS]]{{.*}}
+// CHECK: [[RANGE_STR:.*]] = {{.*}}_NSRange=II{{.*}}
+// CHECK: [[METH:@.*]] = private global{{.*}}valueWithBytes:objCType:{{.*}}
+// CHECK: [[VALUE_SEL:@.*]] = {{.*}}[[METH]]{{.*}}
+// CHECK: [[POINT_STR:.*]] = {{.*}}CGPoint=dd{{.*}}
+// CHECK: [[SIZE_STR:.*]] = {{.*}}CGSize=dd{{.*}}
+// CHECK: [[RECT_STR:.*]] = {{.*}}CGRect={CGPoint=dd}{CGSize=dd}}{{.*}}
+// CHECK: [[EDGE_STR:.*]] = {{.*}}NSEdgeInsets=dddd{{.*}}
+
+// CHECK-LABEL: define void @doRange()
+void doRange() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRange* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSRange ns_range = { .location = 0, .length = 42 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *range = @(ns_range);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doPoint()
+void doPoint() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGPoint{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGPoint{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGPoint* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGPoint cg_point = { .x = 42, .y = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[POINT_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *point = @(cg_point);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doSize()
+void doSize() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGSize{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGSize{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGSize* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGSize cg_size = { .width = 42, .height = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[SIZE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *size = @(cg_size);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRect()
+void doRect() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGRect{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGRect{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGRect* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGPoint cg_point = { .x = 42, .y = 24 };
+ CGSize cg_size = { .width = 42, .height = 24 };
+ CGRect cg_rect = { .origin = cg_point, .size = cg_size };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[RECT_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *rect = @(cg_rect);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doNSEdgeInsets()
+void doNSEdgeInsets() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSEdgeInsets ns_edge_insets;
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[EDGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *edge_insets = @(ns_edge_insets);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRangeRValue()
+void doRangeRValue() {
+ // CHECK: [[COERCE:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: call {{.*}} @getRange {{.*}} [[COERCE]]{{.*}}
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[COERCE_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *range_rvalue = @(getRange());
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
diff --git a/test/CodeGenObjC/nsvalue-objc-boxable-ios.m b/test/CodeGenObjC/nsvalue-objc-boxable-ios.m
new file mode 100644
index 000000000000..3443babdc9d0
--- /dev/null
+++ b/test/CodeGenObjC/nsvalue-objc-boxable-ios.m
@@ -0,0 +1,114 @@
+// RUN: %clang_cc1 -I %S/Inputs -triple armv7-apple-ios8.0.0 -emit-llvm -O2 -disable-llvm-optzns -o - %s | FileCheck %s
+
+#import "nsvalue-boxed-expressions-support.h"
+
+// CHECK: [[CLASS:@.*]] = external global %struct._class_t
+// CHECK: [[NSVALUE:@.*]] = {{.*}}[[CLASS]]{{.*}}
+// CHECK: [[RANGE_STR:.*]] = {{.*}}_NSRange=II{{.*}}
+// CHECK: [[METH:@.*]] = private global{{.*}}valueWithBytes:objCType:{{.*}}
+// CHECK: [[VALUE_SEL:@.*]] = {{.*}}[[METH]]{{.*}}
+// CHECK: [[POINT_STR:.*]] = {{.*}}CGPoint=dd{{.*}}
+// CHECK: [[SIZE_STR:.*]] = {{.*}}CGSize=dd{{.*}}
+// CHECK: [[RECT_STR:.*]] = {{.*}}CGRect={CGPoint=dd}{CGSize=dd}}{{.*}}
+// CHECK: [[EDGE_STR:.*]] = {{.*}}NSEdgeInsets=dddd{{.*}}
+
+// CHECK-LABEL: define void @doRange()
+void doRange() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRange* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSRange ns_range = { .location = 0, .length = 42 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ NSValue *range = @(ns_range);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doPoint()
+void doPoint() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGPoint{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGPoint{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGPoint* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGPoint cg_point = { .x = 42, .y = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[POINT_STR]]{{.*}})
+ NSValue *point = @(cg_point);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doSize()
+void doSize() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGSize{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGSize{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGSize* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGSize cg_size = { .width = 42, .height = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[SIZE_STR]]{{.*}})
+ NSValue *size = @(cg_size);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRect()
+void doRect() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.CGRect{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.CGRect{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.CGRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.CGRect* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.CGRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ CGPoint cg_point = { .x = 42, .y = 24 };
+ CGSize cg_size = { .width = 42, .height = 24 };
+ CGRect cg_rect = { .origin = cg_point, .size = cg_size };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[RECT_STR]]{{.*}})
+ NSValue *rect = @(cg_rect);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doNSEdgeInsets()
+void doNSEdgeInsets() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSEdgeInsets ns_edge_insets;
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[EDGE_STR]]{{.*}})
+ NSValue *edge_insets = @(ns_edge_insets);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRangeRValue()
+void doRangeRValue() {
+ // CHECK: [[COERCE:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: call {{.*}} @getRange {{.*}} [[COERCE]]{{.*}}
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[COERCE_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ NSValue *range_rvalue = @(getRange());
+ // CHECK: ret void
+}
+
diff --git a/test/CodeGenObjC/nsvalue-objc-boxable-mac-arc.m b/test/CodeGenObjC/nsvalue-objc-boxable-mac-arc.m
new file mode 100644
index 000000000000..92c91d7d3810
--- /dev/null
+++ b/test/CodeGenObjC/nsvalue-objc-boxable-mac-arc.m
@@ -0,0 +1,130 @@
+// RUN: %clang_cc1 -I %S/Inputs -triple x86_64-apple-macosx -emit-llvm -fobjc-arc -O2 -disable-llvm-optzns -o - %s | FileCheck %s
+
+#import "nsvalue-boxed-expressions-support.h"
+
+// CHECK: [[CLASS:@.*]] = external global %struct._class_t
+// CHECK: [[NSVALUE:@.*]] = {{.*}}[[CLASS]]{{.*}}
+// CHECK: [[RANGE_STR:.*]] = {{.*}}_NSRange=QQ{{.*}}
+// CHECK: [[METH:@.*]] = private global{{.*}}valueWithBytes:objCType:{{.*}}
+// CHECK: [[VALUE_SEL:@.*]] = {{.*}}[[METH]]{{.*}}
+// CHECK: [[POINT_STR:.*]] = {{.*}}_NSPoint=dd{{.*}}
+// CHECK: [[SIZE_STR:.*]] = {{.*}}_NSSize=dd{{.*}}
+// CHECK: [[RECT_STR:.*]] = {{.*}}_NSRect={_NSPoint=dd}{_NSSize=dd}}{{.*}}
+// CHECK: [[EDGE_STR:.*]] = {{.*}}NSEdgeInsets=dddd{{.*}}
+
+// CHECK-LABEL: define void @doRange()
+void doRange() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRange* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSRange ns_range = { .location = 0, .length = 42 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *range = @(ns_range);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doPoint()
+void doPoint() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSPoint{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSPoint{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSPoint* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSPoint ns_point = { .x = 42, .y = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[POINT_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *point = @(ns_point);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doSize()
+void doSize() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSSize{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSSize{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSSize* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSSize ns_size = { .width = 42, .height = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[SIZE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *size = @(ns_size);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRect()
+void doRect() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRect{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRect{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRect* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSPoint ns_point = { .x = 42, .y = 24 };
+ NSSize ns_size = { .width = 42, .height = 24 };
+ NSRect ns_rect = { .origin = ns_point, .size = ns_size };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[RECT_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *rect = @(ns_rect);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doNSEdgeInsets()
+void doNSEdgeInsets() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSEdgeInsets ns_edge_insets;
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[EDGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *edge_insets = @(ns_edge_insets);
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRangeRValue()
+void doRangeRValue() {
+ // CHECK: [[COERCE:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[RVAL:%.*]] = call {{.*}} @getRange()
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[COERCE_CAST_PTR:%.*]] = getelementptr {{.*}} [[COERCE_CAST]], {{.*}}
+ // CHECK: [[EXTR_RVAL:%.*]] = extractvalue {{.*}} [[RVAL]]{{.*}}
+ // CHECK: store {{.*}}[[EXTR_RVAL]]{{.*}}[[COERCE_CAST_PTR]]{{.*}}
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[COERCE_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ // CHECK: call i8* @objc_retainAutoreleasedReturnValue
+ NSValue *range_rvalue = @(getRange());
+ // CHECK: call void @objc_release
+ // CHECK: ret void
+}
+
diff --git a/test/CodeGenObjC/nsvalue-objc-boxable-mac.m b/test/CodeGenObjC/nsvalue-objc-boxable-mac.m
new file mode 100644
index 000000000000..da0ae7a57108
--- /dev/null
+++ b/test/CodeGenObjC/nsvalue-objc-boxable-mac.m
@@ -0,0 +1,118 @@
+// RUN: %clang_cc1 -I %S/Inputs -triple x86_64-apple-macosx -emit-llvm -O2 -disable-llvm-optzns -o - %s | FileCheck %s
+
+#import "nsvalue-boxed-expressions-support.h"
+
+// CHECK: [[CLASS:@.*]] = external global %struct._class_t
+// CHECK: [[NSVALUE:@.*]] = {{.*}}[[CLASS]]{{.*}}
+// CHECK: [[RANGE_STR:.*]] = {{.*}}_NSRange=QQ{{.*}}
+// CHECK: [[METH:@.*]] = private global{{.*}}valueWithBytes:objCType:{{.*}}
+// CHECK: [[VALUE_SEL:@.*]] = {{.*}}[[METH]]{{.*}}
+// CHECK: [[POINT_STR:.*]] = {{.*}}_NSPoint=dd{{.*}}
+// CHECK: [[SIZE_STR:.*]] = {{.*}}_NSSize=dd{{.*}}
+// CHECK: [[RECT_STR:.*]] = {{.*}}_NSRect={_NSPoint=dd}{_NSSize=dd}}{{.*}}
+// CHECK: [[EDGE_STR:.*]] = {{.*}}NSEdgeInsets=dddd{{.*}}
+
+// CHECK-LABEL: define void @doRange()
+void doRange() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRange* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRange* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSRange ns_range = { .location = 0, .length = 42 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ NSValue *range = @(ns_range);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doPoint()
+void doPoint() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSPoint{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSPoint{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSPoint* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSPoint* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSPoint ns_point = { .x = 42, .y = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[POINT_STR]]{{.*}})
+ NSValue *point = @(ns_point);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doSize()
+void doSize() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSSize{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSSize{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSSize* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSSize* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSSize ns_size = { .width = 42, .height = 24 };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8* {{.*}}[[SIZE_STR]]{{.*}})
+ NSValue *size = @(ns_size);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRect()
+void doRect() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct._NSRect{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct._NSRect{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct._NSRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct._NSRect* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct._NSRect* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSPoint ns_point = { .x = 42, .y = 24 };
+ NSSize ns_size = { .width = 42, .height = 24 };
+ NSRect ns_rect = { .origin = ns_point, .size = ns_size };
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[RECT_STR]]{{.*}})
+ NSValue *rect = @(ns_rect);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doNSEdgeInsets()
+void doNSEdgeInsets() {
+ // CHECK: [[LOCAL_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[TEMP_VAR:%.*]] = alloca %struct.NSEdgeInsets{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[TEMP_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[LOCAL_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[LOCAL_VAR]]{{.*}}
+ // CHECK: call void @llvm.memcpy{{.*}} [[TEMP_CAST]]{{.*}} [[LOCAL_CAST]]{{.*}}
+ // CHECK: [[PARAM_CAST:%.*]] = bitcast %struct.NSEdgeInsets* [[TEMP_VAR]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ NSEdgeInsets ns_edge_insets;
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[PARAM_CAST]], i8*{{.*}}[[EDGE_STR]]{{.*}})
+ NSValue *edge_insets = @(ns_edge_insets);
+ // CHECK: ret void
+}
+
+// CHECK-LABEL: define void @doRangeRValue()
+void doRangeRValue() {
+ // CHECK: [[COERCE:%.*]] = alloca %struct._NSRange{{.*}}
+ // CHECK: [[RECV_PTR:%.*]] = load {{.*}} [[NSVALUE]]
+ // CHECK: [[RVAL:%.*]] = call {{.*}} @getRange()
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[COERCE_CAST_PTR:%.*]] = getelementptr {{.*}} [[COERCE_CAST]], {{.*}}
+ // CHECK: [[EXTR_RVAL:%.*]] = extractvalue {{.*}} [[RVAL]]{{.*}}
+ // CHECK: store {{.*}}[[EXTR_RVAL]]{{.*}}[[COERCE_CAST_PTR]]{{.*}}
+ // CHECK: [[COERCE_CAST:%.*]] = bitcast %struct._NSRange* [[COERCE]]{{.*}}
+ // CHECK: [[SEL:%.*]] = load i8*, i8** [[VALUE_SEL]]
+ // CHECK: [[RECV:%.*]] = bitcast %struct._class_t* [[RECV_PTR]] to i8*
+ // CHECK: call {{.*objc_msgSend.*}}(i8* [[RECV]], i8* [[SEL]], i8* [[COERCE_CAST]], i8* {{.*}}[[RANGE_STR]]{{.*}})
+ NSValue *range_rvalue = @(getRange());
+ // CHECK: ret void
+}
+