1*67e74705SXin Li // RUN: %clang_cc1 -triple x86_64-unknown-freebsd10.0 -emit-llvm < %s | FileCheck -check-prefix=FREEBSD %s
2*67e74705SXin Li // RUN: %clang_cc1 -triple x86_64-pc-win32 -emit-llvm < %s | FileCheck -check-prefix=WIN64 %s
3*67e74705SXin Li
4*67e74705SXin Li struct foo {
5*67e74705SXin Li int x;
6*67e74705SXin Li float y;
7*67e74705SXin Li char z;
8*67e74705SXin Li };
9*67e74705SXin Li // FREEBSD: %[[STRUCT_FOO:.*]] = type { i32, float, i8 }
10*67e74705SXin Li // WIN64: %[[STRUCT_FOO:.*]] = type { i32, float, i8 }
11*67e74705SXin Li
12*67e74705SXin Li void __attribute__((ms_abi)) f1(void);
13*67e74705SXin Li void __attribute__((sysv_abi)) f2(void);
f3(void)14*67e74705SXin Li void f3(void) {
15*67e74705SXin Li // FREEBSD-LABEL: define void @f3()
16*67e74705SXin Li // WIN64-LABEL: define void @f3()
17*67e74705SXin Li f1();
18*67e74705SXin Li // FREEBSD: call x86_64_win64cc void @f1()
19*67e74705SXin Li // WIN64: call void @f1()
20*67e74705SXin Li f2();
21*67e74705SXin Li // FREEBSD: call void @f2()
22*67e74705SXin Li // WIN64: call x86_64_sysvcc void @f2()
23*67e74705SXin Li }
24*67e74705SXin Li // FREEBSD: declare x86_64_win64cc void @f1()
25*67e74705SXin Li // FREEBSD: declare void @f2()
26*67e74705SXin Li // WIN64: declare void @f1()
27*67e74705SXin Li // WIN64: declare x86_64_sysvcc void @f2()
28*67e74705SXin Li
29*67e74705SXin Li // Win64 ABI varargs
f4(int a,...)30*67e74705SXin Li void __attribute__((ms_abi)) f4(int a, ...) {
31*67e74705SXin Li // FREEBSD-LABEL: define x86_64_win64cc void @f4
32*67e74705SXin Li // WIN64-LABEL: define void @f4
33*67e74705SXin Li __builtin_ms_va_list ap;
34*67e74705SXin Li __builtin_ms_va_start(ap, a);
35*67e74705SXin Li // FREEBSD: %[[AP:.*]] = alloca i8*
36*67e74705SXin Li // FREEBSD: call void @llvm.va_start
37*67e74705SXin Li // WIN64: %[[AP:.*]] = alloca i8*
38*67e74705SXin Li // WIN64: call void @llvm.va_start
39*67e74705SXin Li int b = __builtin_va_arg(ap, int);
40*67e74705SXin Li // FREEBSD: %[[AP_CUR:.*]] = load i8*, i8** %[[AP]]
41*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR]], i64 8
42*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT]], i8** %[[AP]]
43*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR]] to i32*
44*67e74705SXin Li // WIN64: %[[AP_CUR:.*]] = load i8*, i8** %[[AP]]
45*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR]], i64 8
46*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT]], i8** %[[AP]]
47*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR]] to i32*
48*67e74705SXin Li double _Complex c = __builtin_va_arg(ap, double _Complex);
49*67e74705SXin Li // FREEBSD: %[[AP_CUR2:.*]] = load i8*, i8** %[[AP]]
50*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT2:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR2]], i64 16
51*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT2]], i8** %[[AP]]
52*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR2]] to { double, double }*
53*67e74705SXin Li // WIN64: %[[AP_CUR2:.*]] = load i8*, i8** %[[AP]]
54*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT2:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR2]], i64 16
55*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT2]], i8** %[[AP]]
56*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR2]] to { double, double }*
57*67e74705SXin Li struct foo d = __builtin_va_arg(ap, struct foo);
58*67e74705SXin Li // FREEBSD: %[[AP_CUR3:.*]] = load i8*, i8** %[[AP]]
59*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT3:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR3]], i64 16
60*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT3]], i8** %[[AP]]
61*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR3]] to %[[STRUCT_FOO]]*
62*67e74705SXin Li // WIN64: %[[AP_CUR3:.*]] = load i8*, i8** %[[AP]]
63*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT3:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR3]], i64 16
64*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT3]], i8** %[[AP]]
65*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR3]] to %[[STRUCT_FOO]]*
66*67e74705SXin Li __builtin_ms_va_list ap2;
67*67e74705SXin Li __builtin_ms_va_copy(ap2, ap);
68*67e74705SXin Li // FREEBSD: %[[AP_VAL:.*]] = load i8*, i8** %[[AP]]
69*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_VAL]], i8** %[[AP2:.*]]
70*67e74705SXin Li // WIN64: %[[AP_VAL:.*]] = load i8*, i8** %[[AP]]
71*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_VAL]], i8** %[[AP2:.*]]
72*67e74705SXin Li __builtin_ms_va_end(ap);
73*67e74705SXin Li // FREEBSD: call void @llvm.va_end
74*67e74705SXin Li // WIN64: call void @llvm.va_end
75*67e74705SXin Li }
76*67e74705SXin Li
77*67e74705SXin Li // Let's verify that normal va_lists work right on Win64, too.
f5(int a,...)78*67e74705SXin Li void f5(int a, ...) {
79*67e74705SXin Li // WIN64-LABEL: define void @f5
80*67e74705SXin Li __builtin_va_list ap;
81*67e74705SXin Li __builtin_va_start(ap, a);
82*67e74705SXin Li // WIN64: %[[AP:.*]] = alloca i8*
83*67e74705SXin Li // WIN64: call void @llvm.va_start
84*67e74705SXin Li int b = __builtin_va_arg(ap, int);
85*67e74705SXin Li // WIN64: %[[AP_CUR:.*]] = load i8*, i8** %[[AP]]
86*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR]], i64 8
87*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT]], i8** %[[AP]]
88*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR]] to i32*
89*67e74705SXin Li double _Complex c = __builtin_va_arg(ap, double _Complex);
90*67e74705SXin Li // WIN64: %[[AP_CUR2:.*]] = load i8*, i8** %[[AP]]
91*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT2:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR2]], i64 16
92*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT2]], i8** %[[AP]]
93*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR2]] to { double, double }*
94*67e74705SXin Li struct foo d = __builtin_va_arg(ap, struct foo);
95*67e74705SXin Li // WIN64: %[[AP_CUR3:.*]] = load i8*, i8** %[[AP]]
96*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT3:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR3]], i64 16
97*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT3]], i8** %[[AP]]
98*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR3]] to %[[STRUCT_FOO]]*
99*67e74705SXin Li __builtin_va_list ap2;
100*67e74705SXin Li __builtin_va_copy(ap2, ap);
101*67e74705SXin Li // WIN64: call void @llvm.va_copy
102*67e74705SXin Li __builtin_va_end(ap);
103*67e74705SXin Li // WIN64: call void @llvm.va_end
104*67e74705SXin Li }
105*67e74705SXin Li
106*67e74705SXin Li // Verify that using a Win64 va_list from a System V function works.
f6(__builtin_ms_va_list ap)107*67e74705SXin Li void __attribute__((sysv_abi)) f6(__builtin_ms_va_list ap) {
108*67e74705SXin Li // FREEBSD-LABEL: define void @f6
109*67e74705SXin Li // FREEBSD: store i8* %ap, i8** %[[AP:.*]]
110*67e74705SXin Li // WIN64-LABEL: define x86_64_sysvcc void @f6
111*67e74705SXin Li // WIN64: store i8* %ap, i8** %[[AP:.*]]
112*67e74705SXin Li int b = __builtin_va_arg(ap, int);
113*67e74705SXin Li // FREEBSD: %[[AP_CUR:.*]] = load i8*, i8** %[[AP]]
114*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR]], i64 8
115*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT]], i8** %[[AP]]
116*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR]] to i32*
117*67e74705SXin Li // WIN64: %[[AP_CUR:.*]] = load i8*, i8** %[[AP]]
118*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR]], i64 8
119*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT]], i8** %[[AP]]
120*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR]] to i32*
121*67e74705SXin Li double _Complex c = __builtin_va_arg(ap, double _Complex);
122*67e74705SXin Li // FREEBSD: %[[AP_CUR2:.*]] = load i8*, i8** %[[AP]]
123*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT2:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR2]], i64 16
124*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT2]], i8** %[[AP]]
125*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR2]] to { double, double }*
126*67e74705SXin Li // WIN64: %[[AP_CUR2:.*]] = load i8*, i8** %[[AP]]
127*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT2:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR2]], i64 16
128*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT2]], i8** %[[AP]]
129*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR2]] to { double, double }*
130*67e74705SXin Li struct foo d = __builtin_va_arg(ap, struct foo);
131*67e74705SXin Li // FREEBSD: %[[AP_CUR3:.*]] = load i8*, i8** %[[AP]]
132*67e74705SXin Li // FREEBSD-NEXT: %[[AP_NEXT3:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR3]], i64 16
133*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_NEXT3]], i8** %[[AP]]
134*67e74705SXin Li // FREEBSD-NEXT: bitcast i8* %[[AP_CUR3]] to %[[STRUCT_FOO]]*
135*67e74705SXin Li // WIN64: %[[AP_CUR3:.*]] = load i8*, i8** %[[AP]]
136*67e74705SXin Li // WIN64-NEXT: %[[AP_NEXT3:.*]] = getelementptr inbounds i8, i8* %[[AP_CUR3]], i64 16
137*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_NEXT3]], i8** %[[AP]]
138*67e74705SXin Li // WIN64-NEXT: bitcast i8* %[[AP_CUR3]] to %[[STRUCT_FOO]]*
139*67e74705SXin Li __builtin_ms_va_list ap2;
140*67e74705SXin Li __builtin_ms_va_copy(ap2, ap);
141*67e74705SXin Li // FREEBSD: %[[AP_VAL:.*]] = load i8*, i8** %[[AP]]
142*67e74705SXin Li // FREEBSD-NEXT: store i8* %[[AP_VAL]], i8** %[[AP2:.*]]
143*67e74705SXin Li // WIN64: %[[AP_VAL:.*]] = load i8*, i8** %[[AP]]
144*67e74705SXin Li // WIN64-NEXT: store i8* %[[AP_VAL]], i8** %[[AP2:.*]]
145*67e74705SXin Li }
146