1*67e74705SXin Li // RUN: %clang_cc1 -emit-llvm -triple=x86_64-pc-win32 -fms-compatibility %s -o - | FileCheck %s
2*67e74705SXin Li // RUN: %clang_cc1 -triple=x86_64-pc-win32 -fms-compatibility -emit-pch -o %t %s
3*67e74705SXin Li // RUN: %clang_cc1 -emit-llvm -triple=x86_64-pc-win32 -fms-compatibility -include-pch %t -verify %s -o - | FileCheck %s
4*67e74705SXin Li // expected-no-diagnostics
5*67e74705SXin Li
6*67e74705SXin Li #ifndef HEADER
7*67e74705SXin Li #define HEADER
8*67e74705SXin Li
9*67e74705SXin Li class Test1 {
10*67e74705SXin Li private:
11*67e74705SXin Li int x_;
12*67e74705SXin Li double y_;
13*67e74705SXin Li
14*67e74705SXin Li public:
Test1(int x)15*67e74705SXin Li Test1(int x) : x_(x) {}
16*67e74705SXin Li __declspec(property(get = get_x)) int X;
get_x() const17*67e74705SXin Li int get_x() const { return x_; }
GetTest1()18*67e74705SXin Li static Test1 *GetTest1() { return new Test1(10); }
19*67e74705SXin Li };
20*67e74705SXin Li
21*67e74705SXin Li class S {
22*67e74705SXin Li public:
23*67e74705SXin Li __declspec(property(get=GetX,put=PutX)) int x[];
GetX(int i,int j)24*67e74705SXin Li int GetX(int i, int j) { return i+j; }
PutX(int i,int j,int k)25*67e74705SXin Li void PutX(int i, int j, int k) { j = i = k; }
26*67e74705SXin Li };
27*67e74705SXin Li
28*67e74705SXin Li template <typename T>
29*67e74705SXin Li class St {
30*67e74705SXin Li public:
31*67e74705SXin Li __declspec(property(get=GetX,put=PutX)) T x[];
GetX(T i,T j)32*67e74705SXin Li T GetX(T i, T j) { return i+j; }
GetX()33*67e74705SXin Li T GetX() { return 0; }
PutX(T i,T j,T k)34*67e74705SXin Li T PutX(T i, T j, T k) { return j = i = k; }
35*67e74705SXin Li __declspec(property(get=GetY,put=PutY)) T y[];
GetY(char i,Test1 j)36*67e74705SXin Li char GetY(char i, Test1 j) { return i+j.get_x(); }
PutY(char i,int j,double k)37*67e74705SXin Li void PutY(char i, int j, double k) { j = i = k; }
38*67e74705SXin Li };
39*67e74705SXin Li
40*67e74705SXin Li template <typename T>
foo(T i,T j)41*67e74705SXin Li void foo(T i, T j) {
42*67e74705SXin Li St<T> bar;
43*67e74705SXin Li Test1 t(i);
44*67e74705SXin Li bar.x[i][j] = bar.x[i][j];
45*67e74705SXin Li bar.y[t.X][j] = bar.x[i][j];
46*67e74705SXin Li bar.x[i][j] = bar.y[bar.x[i][j]][t];
47*67e74705SXin Li }
48*67e74705SXin Li
idx()49*67e74705SXin Li int idx() { return 7; }
50*67e74705SXin Li
51*67e74705SXin Li // CHECK-LABEL: main
main(int argc,char ** argv)52*67e74705SXin Li int main(int argc, char **argv) {
53*67e74705SXin Li Test1 t(argc);
54*67e74705SXin Li S *p1 = 0;
55*67e74705SXin Li St<float> *p2 = 0;
56*67e74705SXin Li // CHECK: call i32 @"\01?GetX@S@@QEAAHHH@Z"(%class.S* %{{.+}}, i32 223, i32 11)
57*67e74705SXin Li int j = p1->x[223][11];
58*67e74705SXin Li // CHECK: [[J:%.+]] = load i32, i32* %
59*67e74705SXin Li // CHECK-NEXT: call void @"\01?PutX@S@@QEAAXHHH@Z"(%class.S* %{{.+}}, i32 23, i32 1, i32 [[J]])
60*67e74705SXin Li p1->x[23][1] = j;
61*67e74705SXin Li // CHECK: call float @"\01?GetX@?$St@M@@QEAAMMM@Z"(%class.St* %{{.+}}, float 2.230000e+02, float 1.100000e+01)
62*67e74705SXin Li float j1 = p2->x[223][11];
63*67e74705SXin Li // CHECK: [[J1:%.+]] = load float, float* %
64*67e74705SXin Li // CHECK-NEXT: [[CALL:%.+]] = call float @"\01?PutX@?$St@M@@QEAAMMMM@Z"(%class.St* %{{.+}}, float 2.300000e+01, float 1.000000e+00, float [[J1]])
65*67e74705SXin Li // CHECK-NEXT: [[CONV:%.+]] = fptosi float [[CALL]] to i32
66*67e74705SXin Li // CHECK-NEXT: store i32 [[CONV]], i32*
67*67e74705SXin Li argc = p2->x[23][1] = j1;
68*67e74705SXin Li // CHECK: [[IDX:%.+]] = call i32 @"\01?idx@@YAHXZ"()
69*67e74705SXin Li // CHECK-NEXT: [[CONV:%.+]] = sitofp i32 [[IDX]] to float
70*67e74705SXin Li // CHECK-NEXT: [[GET:%.+]] = call float @"\01?GetX@?$St@M@@QEAAMMM@Z"(%class.St* %{{.+}}, float [[CONV]], float 1.000000e+00)
71*67e74705SXin Li // CHECK-NEXT: [[INC:%.+]] = fadd float [[GET]], 1.000000e+00
72*67e74705SXin Li // CHECK-NEXT: [[CONV:%.+]] = sitofp i32 [[IDX]] to float
73*67e74705SXin Li // CHECK-NEXT: call float @"\01?PutX@?$St@M@@QEAAMMMM@Z"(%class.St* %{{.+}}, float [[CONV]], float 1.000000e+00, float [[INC]])
74*67e74705SXin Li ++p2->x[idx()][1];
75*67e74705SXin Li // CHECK: call void @"\01??$foo@H@@YAXHH@Z"(i32 %{{.+}}, i32 %{{.+}})
76*67e74705SXin Li foo(argc, (int)argv[0][0]);
77*67e74705SXin Li // CHECK: [[P2:%.+]] = load %class.St*, %class.St** %
78*67e74705SXin Li // CHECK: [[T_X:%.+]] = call i32 @"\01?get_x@Test1@@QEBAHXZ"(%class.Test1* %{{.+}})
79*67e74705SXin Li // CHECK: [[P1:%.+]] = load %class.S*, %class.S** %
80*67e74705SXin Li // CHECK: [[P1_X_22_33:%.+]] = call i32 @"\01?GetX@S@@QEAAHHH@Z"(%class.S* [[P1]], i32 22, i32 33)
81*67e74705SXin Li // CHECK: [[CAST:%.+]] = sitofp i32 [[P1_X_22_33]] to double
82*67e74705SXin Li // CHECK: [[ARGC:%.+]] = load i32, i32* %
83*67e74705SXin Li // CHECK: [[CAST2:%.+]] = trunc i32 [[T_X]] to i8
84*67e74705SXin Li // CHECK: call void @"\01?PutY@?$St@M@@QEAAXDHN@Z"(%class.St* [[P2]], i8 [[CAST2]], i32 [[ARGC]], double [[CAST]])
85*67e74705SXin Li p2->y[t.X][argc] = p1->x[22][33];
86*67e74705SXin Li // CHECK: [[P2_1:%.+]] = load %class.St*, %class.St**
87*67e74705SXin Li // CHECK: [[P2_2:%.+]] = load %class.St*, %class.St**
88*67e74705SXin Li // CHECK: [[P1:%.+]] = load %class.S*, %class.S**
89*67e74705SXin Li // CHECK: [[ARGC:%.+]] = load i32, i32* %
90*67e74705SXin Li // CHECK: [[P1_X_ARGC_0:%.+]] = call i32 @"\01?GetX@S@@QEAAHHH@Z"(%class.S* [[P1]], i32 [[ARGC]], i32 0)
91*67e74705SXin Li // CHECK: [[CAST:%.+]] = trunc i32 [[P1_X_ARGC_0]] to i8
92*67e74705SXin Li // CHECK: [[P2_Y_p1_X_ARGC_0_T:%.+]] = call i8 @"\01?GetY@?$St@M@@QEAADDVTest1@@@Z"(%class.St* [[P2_2]], i8 [[CAST]], %class.Test1* %{{.+}})
93*67e74705SXin Li // CHECK: [[CAST:%.+]] = sitofp i8 [[P2_Y_p1_X_ARGC_0_T]] to float
94*67e74705SXin Li // CHECK: [[J:%.+]] = load i32, i32* %
95*67e74705SXin Li // CHECK: [[CAST1:%.+]] = sitofp i32 [[J]] to float
96*67e74705SXin Li // CHECK: [[J:%.+]] = load i32, i32* %
97*67e74705SXin Li // CHECK: [[CAST2:%.+]] = sitofp i32 [[J]] to float
98*67e74705SXin Li // CHECK: call float @"\01?PutX@?$St@M@@QEAAMMMM@Z"(%class.St* [[P2_1]], float [[CAST2]], float [[CAST1]], float [[CAST]])
99*67e74705SXin Li p2->x[j][j] = p2->y[p1->x[argc][0]][t];
100*67e74705SXin Li // CHECK: [[CALL:%.+]] = call %class.Test1* @"\01?GetTest1@Test1@@SAPEAV1@XZ"()
101*67e74705SXin Li // CHECK-NEXT: call i32 @"\01?get_x@Test1@@QEBAHXZ"(%class.Test1* [[CALL]])
102*67e74705SXin Li return Test1::GetTest1()->X;
103*67e74705SXin Li }
104*67e74705SXin Li
105*67e74705SXin Li // CHECK: define linkonce_odr void @"\01??$foo@H@@YAXHH@Z"(i32 %{{.+}}, i32 %{{.+}})
106*67e74705SXin Li // CHECK: call i32 @"\01?GetX@?$St@H@@QEAAHHH@Z"(%class.St{{.+}}* [[BAR:%.+]], i32 %{{.+}} i32 %{{.+}})
107*67e74705SXin Li // CHECK: call i32 @"\01?PutX@?$St@H@@QEAAHHHH@Z"(%class.St{{.+}}* [[BAR]], i32 %{{.+}}, i32 %{{.+}}, i32 %{{.+}})
108*67e74705SXin Li // CHECK: call i32 @"\01?GetX@?$St@H@@QEAAHHH@Z"(%class.St{{.+}}* [[BAR]], i32 %{{.+}} i32 %{{.+}})
109*67e74705SXin Li // CHECK: call void @"\01?PutY@?$St@H@@QEAAXDHN@Z"(%class.St{{.+}}* [[BAR]], i8 %{{.+}}, i32 %{{.+}}, double %{{.+}}
110*67e74705SXin Li // CHECK: call i32 @"\01?GetX@?$St@H@@QEAAHHH@Z"(%class.St{{.+}}* [[BAR]], i32 %{{.+}} i32 %{{.+}})
111*67e74705SXin Li // CHECK: call i8 @"\01?GetY@?$St@H@@QEAADDVTest1@@@Z"(%class.St{{.+}}* [[BAR]], i8 %{{.+}}, %class.Test1* %{{.+}})
112*67e74705SXin Li // CHECK: call i32 @"\01?PutX@?$St@H@@QEAAHHHH@Z"(%class.St{{.+}}* [[BAR]], i32 %{{.+}}, i32 %{{.+}}, i32 %{{.+}})
113*67e74705SXin Li #endif //HEADER
114