1 | // RUN: %clang_cc1 -fsyntax-only -verify %s |
2 | |
3 | namespace test1 { |
4 | extern "C" { |
5 | void test1_f() { |
6 | void test1_g(int); |
7 | } |
8 | } |
9 | } |
10 | int test1_g(int); |
11 | |
12 | namespace test2 { |
13 | extern "C" { |
14 | void test2_f() { |
15 | extern int test2_x; // expected-note {{declared with C language linkage here}} |
16 | } |
17 | } |
18 | } |
19 | float test2_x; // expected-error {{declaration of 'test2_x' in global scope conflicts with declaration with C language linkage}} |
20 | |
21 | namespace test3 { |
22 | extern "C" { |
23 | void test3_f() { |
24 | extern int test3_b; // expected-note {{previous declaration is here}} |
25 | } |
26 | } |
27 | extern "C" { |
28 | float test3_b; // expected-error {{redefinition of 'test3_b' with a different type: 'float' vs 'int'}} |
29 | } |
30 | } |
31 | |
32 | namespace N { |
33 | extern "C" { |
34 | void test4_f() { |
35 | extern int test4_b; // expected-note {{declared with C language linkage here}} |
36 | } |
37 | } |
38 | } |
39 | static float test4_b; // expected-error {{declaration of 'test4_b' in global scope conflicts with declaration with C language linkage}} |
40 | |
41 | extern "C" { |
42 | void test4c_f() { |
43 | extern int test4_c; // expected-note {{previous}} |
44 | } |
45 | } |
46 | static float test4_c; // expected-error {{redefinition of 'test4_c' with a different type: 'float' vs 'int'}} |
47 | |
48 | namespace N { |
49 | extern "C" { |
50 | void test5_f() { |
51 | extern int test5_b; // expected-note {{declared with C language linkage here}} |
52 | } |
53 | } |
54 | } |
55 | extern "C" { |
56 | static float test5_b; // expected-error {{declaration of 'test5_b' in global scope conflicts with declaration with C language linkage}} |
57 | } |
58 | |
59 | extern "C" { |
60 | void test5c_f() { |
61 | extern int test5_c; // expected-note {{previous}} |
62 | } |
63 | } |
64 | extern "C" { |
65 | static float test5_c; // expected-error {{redefinition of 'test5_c' with a different type: 'float' vs 'int'}} |
66 | } |
67 | |
68 | extern "C" { |
69 | void f() { |
70 | extern int test6_b; |
71 | } |
72 | } |
73 | namespace foo { |
74 | extern "C" { |
75 | static float test6_b; |
76 | extern float test6_b; |
77 | } |
78 | } |
79 | |
80 | namespace linkage { |
81 | namespace redecl { |
82 | extern "C" { |
83 | static void linkage_redecl(); |
84 | static void linkage_redecl(int); |
85 | void linkage_redecl(); // ok, still not extern "C" |
86 | void linkage_redecl(int); // ok, still not extern "C" |
87 | void linkage_redecl(float); // expected-note {{previous}} |
88 | void linkage_redecl(double); // expected-error {{conflicting types}} |
89 | } |
90 | } |
91 | namespace from_outer { |
92 | void linkage_from_outer_1(); // expected-note {{previous}} |
93 | void linkage_from_outer_2(); // expected-note {{previous}} |
94 | extern "C" { |
95 | void linkage_from_outer_1(int); |
96 | void linkage_from_outer_1(); // expected-error {{different language linkage}} |
97 | void linkage_from_outer_2(); // expected-error {{different language linkage}} |
98 | } |
99 | } |
100 | namespace mixed { |
101 | extern "C" { |
102 | void linkage_mixed_1(); |
103 | static void linkage_mixed_1(int); |
104 | |
105 | static void linkage_mixed_2(int); |
106 | void linkage_mixed_2(); |
107 | } |
108 | } |
109 | namespace across_scopes { |
110 | namespace X { |
111 | extern "C" void linkage_across_scopes_f() { |
112 | void linkage_across_scopes_g(); // expected-note {{previous}} |
113 | } |
114 | } |
115 | namespace Y { |
116 | extern "C" void linkage_across_scopes_g(int); // expected-error {{conflicting}} |
117 | } |
118 | } |
119 | } |
120 | |
121 | int lookup_in_global_f; // expected-note {{here}} |
122 | namespace lookup_in_global { |
123 | void lookup_in_global_f(); |
124 | void lookup_in_global_g(); |
125 | extern "C" { |
126 | void lookup_in_global_f(int); // expected-error {{conflicts with declaration in global scope}} |
127 | void lookup_in_global_g(int); // expected-note {{here}} |
128 | } |
129 | } |
130 | int lookup_in_global_g; // expected-error {{conflicts with declaration with C language linkage}} |
131 | |
132 | namespace N1 { |
133 | extern "C" int different_kind_1; // expected-note {{here}} |
134 | extern "C" void different_kind_2(); // expected-note {{here}} |
135 | } |
136 | namespace N2 { |
137 | extern "C" void different_kind_1(); // expected-error {{different kind of symbol}} |
138 | extern "C" int different_kind_2; // expected-error {{different kind of symbol}} |
139 | } |
140 | |
141 | // We allow all these even though the standard says they are ill-formed. |
142 | extern "C" { |
143 | struct stat {}; // expected-warning{{empty struct has size 0 in C, size 1 in C++}} |
144 | void stat(struct stat); |
145 | } |
146 | namespace X { |
147 | extern "C" { |
148 | void stat(struct ::stat); |
149 | } |
150 | } |
151 | int stat(int *p); |
152 | void global_fn_vs_extern_c_var_1(); |
153 | namespace X { |
154 | extern "C" int global_fn_vs_extern_c_var_1; |
155 | extern "C" int global_fn_vs_extern_c_var_2; |
156 | } |
157 | void global_fn_vs_extern_c_var_2(); |
158 | void global_fn_vs_extern_c_fn_1(); |
159 | namespace X { |
160 | extern "C" int global_fn_vs_extern_c_fn_1(int); |
161 | extern "C" int global_fn_vs_extern_c_fn_2(int); |
162 | } |
163 | void global_fn_vs_extern_c_fn_2(); |
164 | extern "C" void name_with_using_decl_1(int); |
165 | namespace using_decl { |
166 | void name_with_using_decl_1(); |
167 | void name_with_using_decl_2(); |
168 | void name_with_using_decl_3(); |
169 | } |
170 | using using_decl::name_with_using_decl_1; |
171 | using using_decl::name_with_using_decl_2; |
172 | extern "C" void name_with_using_decl_2(int); |
173 | extern "C" void name_with_using_decl_3(int); |
174 | using using_decl::name_with_using_decl_3; |
175 | |
176 | // We do not allow a global variable and an extern "C" function to have the same |
177 | // name, because such entities may have the same mangled name. |
178 | int global_var_vs_extern_c_fn_1; // expected-note {{here}} |
179 | namespace X { |
180 | extern "C" void global_var_vs_extern_c_fn_1(); // expected-error {{conflicts with declaration in global scope}} |
181 | extern "C" void global_var_vs_extern_c_fn_2(); // expected-note {{here}} |
182 | } |
183 | int global_var_vs_extern_c_fn_2; // expected-error {{conflicts with declaration with C language linkage}} |
184 | int global_var_vs_extern_c_var_1; // expected-note {{here}} |
185 | namespace X { |
186 | extern "C" double global_var_vs_extern_c_var_1; // expected-error {{conflicts with declaration in global scope}} |
187 | extern "C" double global_var_vs_extern_c_var_2; // expected-note {{here}} |
188 | } |
189 | int global_var_vs_extern_c_var_2; // expected-error {{conflicts with declaration with C language linkage}} |
190 | |
191 | template <class T> struct pr5065_n1 {}; |
192 | extern "C" { |
193 | union pr5065_1 {}; // expected-warning{{empty union has size 0 in C, size 1 in C++}} |
194 | struct pr5065_2 { int: 0; }; // expected-warning{{struct has size 0 in C, size 1 in C++}} |
195 | struct pr5065_3 {}; // expected-warning{{empty struct has size 0 in C, size 1 in C++}} |
196 | struct pr5065_4 { // expected-warning{{empty struct has size 0 in C, size 1 in C++}} |
197 | struct Inner {}; // expected-warning{{empty struct has size 0 in C, size 1 in C++}} |
198 | }; |
199 | // These should not warn |
200 | class pr5065_n3 {}; |
201 | pr5065_n1<int> pr5065_v; |
202 | struct pr5065_n4 { void m() {} }; |
203 | struct pr5065_n5 : public pr5065_3 {}; |
204 | struct pr5065_n6 : public virtual pr5065_3 {}; |
205 | } |
206 | struct pr5065_n7 {}; |
207 | |
208 | namespace tag_hiding { |
209 | namespace namespace_with_injected_name { |
210 | class Boo { |
211 | friend struct ExternCStruct1; |
212 | }; |
213 | void ExternCStruct4(); // expected-note 2{{candidate}} |
214 | } |
215 | |
216 | class Baz { |
217 | friend struct ExternCStruct2; |
218 | friend void ExternCStruct3(); |
219 | }; |
220 | |
221 | using namespace namespace_with_injected_name; |
222 | |
223 | extern "C" { |
224 | struct ExternCStruct1; |
225 | struct ExternCStruct2; |
226 | struct ExternCStruct3; |
227 | struct ExternCStruct4; // expected-note {{candidate}} |
228 | } |
229 | ExternCStruct1 *p1; |
230 | ExternCStruct2 *p2; |
231 | ExternCStruct3 *p3; |
232 | ExternCStruct4 *p4; // expected-error {{ambiguous}} |
233 | |
234 | extern "C" { |
235 | struct ExternCStruct1; |
236 | struct ExternCStruct2; |
237 | struct ExternCStruct3; |
238 | struct ExternCStruct4; // expected-note {{candidate}} |
239 | } |
240 | ExternCStruct1 *q1 = p1; |
241 | ExternCStruct2 *q2 = p2; |
242 | ExternCStruct3 *q3 = p3; |
243 | ExternCStruct4 *q4 = p4; // expected-error {{ambiguous}} |
244 | } |
245 | |
246 | namespace PR35697 { |
247 | typedef struct {} *ReturnStruct; |
248 | extern "C" ReturnStruct PR35697_f(); |
249 | extern "C" ReturnStruct PR35697_v; |
250 | ReturnStruct PR35697_f(); |
251 | ReturnStruct PR35697_v; |
252 | |
253 | namespace { |
254 | extern "C" ReturnStruct PR35697_f(); |
255 | extern "C" ReturnStruct PR35697_v; |
256 | void q() { |
257 | extern ReturnStruct PR35697_f(); |
258 | extern ReturnStruct PR35697_v; |
259 | } |
260 | } |
261 | } |
262 | |