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