xref: /aosp_15_r20/external/clang/test/SemaCXX/enum-scoped.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -fsyntax-only -pedantic -std=c++11 -verify -triple x86_64-apple-darwin %s
2*67e74705SXin Li 
3*67e74705SXin Li enum class E1 {
4*67e74705SXin Li   Val1 = 1L
5*67e74705SXin Li };
6*67e74705SXin Li 
7*67e74705SXin Li enum struct E2 {
8*67e74705SXin Li   Val1 = '\0'
9*67e74705SXin Li };
10*67e74705SXin Li 
11*67e74705SXin Li E1 v1 = Val1; // expected-error{{undeclared identifier}}
12*67e74705SXin Li E1 v2 = E1::Val1;
13*67e74705SXin Li 
14*67e74705SXin Li static_assert(sizeof(E1) == sizeof(int), "bad size");
15*67e74705SXin Li static_assert(sizeof(E1::Val1) == sizeof(int), "bad size");
16*67e74705SXin Li static_assert(sizeof(E2) == sizeof(int), "bad size");
17*67e74705SXin Li static_assert(sizeof(E2::Val1) == sizeof(int), "bad size");
18*67e74705SXin Li 
19*67e74705SXin Li E1 v3 = E2::Val1; // expected-error{{cannot initialize a variable}}
20*67e74705SXin Li int x1 = E1::Val1; // expected-error{{cannot initialize a variable}}
21*67e74705SXin Li 
22*67e74705SXin Li enum E3 : char {
23*67e74705SXin Li   Val2 = 1
24*67e74705SXin Li };
25*67e74705SXin Li 
26*67e74705SXin Li E3 v4 = Val2;
27*67e74705SXin Li E1 v5 = Val2; // expected-error{{cannot initialize a variable}}
28*67e74705SXin Li 
29*67e74705SXin Li static_assert(sizeof(E3) == 1, "bad size");
30*67e74705SXin Li 
31*67e74705SXin Li int x2 = Val2;
32*67e74705SXin Li 
33*67e74705SXin Li int a1[Val2];
34*67e74705SXin Li int a2[E1::Val1]; // expected-error{{size of array has non-integer type}}
35*67e74705SXin Li 
36*67e74705SXin Li int* p1 = new int[Val2];
37*67e74705SXin Li int* p2 = new int[E1::Val1]; // expected-error{{array size expression must have integral or unscoped enumeration type, not 'E1'}}
38*67e74705SXin Li 
39*67e74705SXin Li enum class E4 {
40*67e74705SXin Li   e1 = -2147483648, // ok
41*67e74705SXin Li   e2 = 2147483647, // ok
42*67e74705SXin Li   e3 = 2147483648 // expected-error{{enumerator value evaluates to 2147483648, which cannot be narrowed to type 'int'}}
43*67e74705SXin Li };
44*67e74705SXin Li 
45*67e74705SXin Li enum class E5 {
46*67e74705SXin Li   e1 = 2147483647, // ok
47*67e74705SXin Li   e2 // expected-error{{2147483648 is not representable in the underlying}}
48*67e74705SXin Li };
49*67e74705SXin Li 
50*67e74705SXin Li enum class E6 : bool {
51*67e74705SXin Li     e1 = false, e2 = true,
52*67e74705SXin Li     e3 // expected-error{{2 is not representable in the underlying}}
53*67e74705SXin Li };
54*67e74705SXin Li 
55*67e74705SXin Li enum E7 : bool {
56*67e74705SXin Li     e1 = false, e2 = true,
57*67e74705SXin Li     e3 // expected-error{{2 is not representable in the underlying}}
58*67e74705SXin Li };
59*67e74705SXin Li 
60*67e74705SXin Li template <class T>
61*67e74705SXin Li struct X {
62*67e74705SXin Li   enum E : T {
63*67e74705SXin Li     e1, e2,
64*67e74705SXin Li     e3 // expected-error{{2 is not representable in the underlying}}
65*67e74705SXin Li   };
66*67e74705SXin Li };
67*67e74705SXin Li 
68*67e74705SXin Li X<bool> X2; // expected-note{{in instantiation of template}}
69*67e74705SXin Li 
70*67e74705SXin Li enum Incomplete1; // expected-error{{C++ forbids forward references}}
71*67e74705SXin Li 
72*67e74705SXin Li enum Complete1 : int;
73*67e74705SXin Li Complete1 complete1;
74*67e74705SXin Li 
75*67e74705SXin Li enum class Complete2;
76*67e74705SXin Li Complete2 complete2;
77*67e74705SXin Li 
78*67e74705SXin Li // All the redeclarations below are done twice on purpose. Tests that the type
79*67e74705SXin Li // of the declaration isn't changed.
80*67e74705SXin Li 
81*67e74705SXin Li enum class Redeclare2; // expected-note{{previous declaration is here}} expected-note{{previous declaration is here}}
82*67e74705SXin Li enum Redeclare2; // expected-error{{previously declared as scoped}}
83*67e74705SXin Li enum Redeclare2; // expected-error{{previously declared as scoped}}
84*67e74705SXin Li 
85*67e74705SXin Li enum Redeclare3 : int; // expected-note{{previous declaration is here}} expected-note{{previous declaration is here}}
86*67e74705SXin Li enum Redeclare3; // expected-error{{previously declared with fixed underlying type}}
87*67e74705SXin Li enum Redeclare3; // expected-error{{previously declared with fixed underlying type}}
88*67e74705SXin Li 
89*67e74705SXin Li enum class Redeclare5;
90*67e74705SXin Li enum class Redeclare5 : int; // ok
91*67e74705SXin Li 
92*67e74705SXin Li enum Redeclare6 : int;   // expected-note{{previous declaration is here}} expected-note{{previous declaration is here}}
93*67e74705SXin Li enum Redeclare6 : short; // expected-error{{redeclared with different underlying type}}
94*67e74705SXin Li enum Redeclare6 : short; // expected-error{{redeclared with different underlying type}}
95*67e74705SXin Li 
96*67e74705SXin Li enum class Redeclare7;         // expected-note{{previous declaration is here}} expected-note{{previous declaration is here}}
97*67e74705SXin Li enum class Redeclare7 : short; // expected-error{{redeclared with different underlying type}}
98*67e74705SXin Li enum class Redeclare7 : short; // expected-error{{redeclared with different underlying type}}
99*67e74705SXin Li 
100*67e74705SXin Li enum : long {
101*67e74705SXin Li   long_enum_val = 10000
102*67e74705SXin Li };
103*67e74705SXin Li 
104*67e74705SXin Li enum : long x; // expected-error{{unnamed enumeration must be a definition}} \
105*67e74705SXin Li // expected-warning{{declaration does not declare anything}}
106*67e74705SXin Li 
PR9333()107*67e74705SXin Li void PR9333() {
108*67e74705SXin Li   enum class scoped_enum { yes, no, maybe };
109*67e74705SXin Li   scoped_enum e = scoped_enum::yes;
110*67e74705SXin Li   if (e == scoped_enum::no) { }
111*67e74705SXin Li }
112*67e74705SXin Li 
113*67e74705SXin Li // <rdar://problem/9366066>
114*67e74705SXin Li namespace rdar9366066 {
115*67e74705SXin Li   enum class X : unsigned { value };
116*67e74705SXin Li 
f(X x)117*67e74705SXin Li   void f(X x) {
118*67e74705SXin Li     x % X::value; // expected-error{{invalid operands to binary expression ('rdar9366066::X' and 'rdar9366066::X')}}
119*67e74705SXin Li     x % 8; // expected-error{{invalid operands to binary expression ('rdar9366066::X' and 'int')}}
120*67e74705SXin Li   }
121*67e74705SXin Li }
122*67e74705SXin Li 
123*67e74705SXin Li // Part 1 of PR10264
124*67e74705SXin Li namespace test5 {
125*67e74705SXin Li   namespace ns {
126*67e74705SXin Li     typedef unsigned Atype;
127*67e74705SXin Li     enum A : Atype;
128*67e74705SXin Li   }
129*67e74705SXin Li   enum ns::A : ns::Atype {
130*67e74705SXin Li     x, y, z
131*67e74705SXin Li   };
132*67e74705SXin Li }
133*67e74705SXin Li 
134*67e74705SXin Li // Part 2 of PR10264
135*67e74705SXin Li namespace test6 {
136*67e74705SXin Li   enum A : unsigned;
137*67e74705SXin Li   struct A::a; // expected-error {{incomplete type 'test6::A' named in nested name specifier}}
138*67e74705SXin Li   enum A::b; // expected-error {{incomplete type 'test6::A' named in nested name specifier}}
139*67e74705SXin Li   int A::c; // expected-error {{incomplete type 'test6::A' named in nested name specifier}}
140*67e74705SXin Li   void A::d(); // expected-error {{incomplete type 'test6::A' named in nested name specifier}}
test()141*67e74705SXin Li   void test() {
142*67e74705SXin Li     (void) A::e; // expected-error {{incomplete type 'test6::A' named in nested name specifier}}
143*67e74705SXin Li   }
144*67e74705SXin Li }
145*67e74705SXin Li 
146*67e74705SXin Li namespace PR11484 {
147*67e74705SXin Li   const int val = 104;
148*67e74705SXin Li   enum class test1 { owner_dead = val, };
149*67e74705SXin Li }
150*67e74705SXin Li 
151*67e74705SXin Li namespace N2764 {
152*67e74705SXin Li   enum class E { a, b };
153*67e74705SXin Li   enum E x1 = E::a; // ok
154*67e74705SXin Li   enum class E x2 = E::a; // expected-error {{reference to scoped enumeration must use 'enum' not 'enum class'}}
155*67e74705SXin Li 
156*67e74705SXin Li   enum F { a, b };
157*67e74705SXin Li   enum F y1 = a; // ok
158*67e74705SXin Li   enum class F y2 = a; // expected-error {{reference to enumeration must use 'enum' not 'enum class'}}
159*67e74705SXin Li 
160*67e74705SXin Li   struct S {
161*67e74705SXin Li     friend enum class E; // expected-error {{reference to scoped enumeration must use 'enum' not 'enum class'}}
162*67e74705SXin Li     friend enum class F; // expected-error {{reference to enumeration must use 'enum' not 'enum class'}}
163*67e74705SXin Li 
164*67e74705SXin Li     friend enum G {}; // expected-error {{forward reference}} expected-error {{cannot define a type in a friend declaration}}
165*67e74705SXin Li     friend enum class H {}; // expected-error {{cannot define a type in a friend declaration}}
166*67e74705SXin Li 
167*67e74705SXin Li     enum A : int;
168*67e74705SXin Li     A a;
169*67e74705SXin Li   } s;
170*67e74705SXin Li 
171*67e74705SXin Li   enum S::A : int {};
172*67e74705SXin Li 
173*67e74705SXin Li   enum class B;
174*67e74705SXin Li }
175*67e74705SXin Li 
176*67e74705SXin Li enum class N2764::B {};
177*67e74705SXin Li 
178*67e74705SXin Li namespace PR12106 {
179*67e74705SXin Li   template<typename E> struct Enum {
EnumPR12106::Enum180*67e74705SXin Li     Enum() : m_e(E::Last) {}
181*67e74705SXin Li     E m_e;
182*67e74705SXin Li   };
183*67e74705SXin Li 
184*67e74705SXin Li   enum eCOLORS { Last };
185*67e74705SXin Li   Enum<eCOLORS> e;
186*67e74705SXin Li }
187*67e74705SXin Li 
188*67e74705SXin Li namespace test7 {
189*67e74705SXin Li   enum class E { e = (struct S*)0 == (struct S*)0 };
190*67e74705SXin Li   S *p;
191*67e74705SXin Li }
192*67e74705SXin Li 
193*67e74705SXin Li namespace test8 {
194*67e74705SXin Li   template<typename T> struct S {
195*67e74705SXin Li     enum A : int; // expected-note {{here}}
196*67e74705SXin Li     enum class B; // expected-note {{here}}
197*67e74705SXin Li     enum class C : int; // expected-note {{here}}
198*67e74705SXin Li     enum class D : int; // expected-note {{here}}
199*67e74705SXin Li   };
200*67e74705SXin Li   template<typename T> enum S<T>::A { a }; // expected-error {{previously declared with fixed underlying type}}
201*67e74705SXin Li   template<typename T> enum class S<T>::B : char { b }; // expected-error {{redeclared with different underlying}}
202*67e74705SXin Li   template<typename T> enum S<T>::C : int { c }; // expected-error {{previously declared as scoped}}
203*67e74705SXin Li   template<typename T> enum class S<T>::D : char { d }; // expected-error {{redeclared with different underlying}}
204*67e74705SXin Li }
205*67e74705SXin Li 
206*67e74705SXin Li namespace test9 {
207*67e74705SXin Li   template<typename T> struct S {
208*67e74705SXin Li     enum class ET : T; // expected-note 2{{here}}
209*67e74705SXin Li     enum class Eint : int; // expected-note 2{{here}}
210*67e74705SXin Li   };
211*67e74705SXin Li   template<> enum class S<int>::ET : int {};
212*67e74705SXin Li   template<> enum class S<char>::ET : short {}; // expected-error {{different underlying type}}
213*67e74705SXin Li   template<> enum class S<int>::Eint : short {}; // expected-error {{different underlying type}}
214*67e74705SXin Li   template<> enum class S<char>::Eint : int {};
215*67e74705SXin Li 
216*67e74705SXin Li   template<typename T> enum class S<T>::ET : int {}; // expected-error {{different underlying type 'int' (was 'short')}}
217*67e74705SXin Li   template<typename T> enum class S<T>::Eint : T {}; // expected-error {{different underlying type 'short' (was 'int')}}
218*67e74705SXin Li 
219*67e74705SXin Li   // The implicit instantiation of S<short> causes the implicit instantiation of
220*67e74705SXin Li   // all declarations of member enumerations, so is ill-formed, even though we
221*67e74705SXin Li   // never instantiate the definitions of S<short>::ET nor S<short>::Eint.
222*67e74705SXin Li   S<short> s; // expected-note {{in instantiation of}}
223*67e74705SXin Li }
224*67e74705SXin Li 
225*67e74705SXin Li namespace test10 {
f()226*67e74705SXin Li   template<typename T> int f() {
227*67e74705SXin Li     enum E : int;
228*67e74705SXin Li     enum E : T; // expected-note {{here}}
229*67e74705SXin Li     E x;
230*67e74705SXin Li     enum E : int { e }; // expected-error {{different underlying}}
231*67e74705SXin Li     x = e;
232*67e74705SXin Li     return x;
233*67e74705SXin Li   }
234*67e74705SXin Li   int k = f<int>();
235*67e74705SXin Li   int l = f<short>(); // expected-note {{here}}
236*67e74705SXin Li 
g()237*67e74705SXin Li   template<typename T> int g() {
238*67e74705SXin Li     enum class E : int;
239*67e74705SXin Li     enum class E : T; // expected-note {{here}}
240*67e74705SXin Li     E x;
241*67e74705SXin Li     enum class E : int { e }; // expected-error {{different underlying}}
242*67e74705SXin Li     x = E::e;
243*67e74705SXin Li     return (int)x;
244*67e74705SXin Li   }
245*67e74705SXin Li   int m = g<int>();
246*67e74705SXin Li   int n = g<short>(); // expected-note {{here}}
247*67e74705SXin Li }
248*67e74705SXin Li 
249*67e74705SXin Li namespace pr13128 {
250*67e74705SXin Li   // This should compile cleanly
251*67e74705SXin Li   class C {
252*67e74705SXin Li     enum class E { C };
253*67e74705SXin Li   };
254*67e74705SXin Li }
255*67e74705SXin Li 
256*67e74705SXin Li namespace PR15633 {
257*67e74705SXin Li   template<typename T> struct A {
258*67e74705SXin Li     struct B {
259*67e74705SXin Li       enum class E : T;
260*67e74705SXin Li       enum class E2 : T;
261*67e74705SXin Li     };
262*67e74705SXin Li   };
263*67e74705SXin Li   template<typename T> enum class A<T>::B::E { e };
264*67e74705SXin Li   template class A<int>;
265*67e74705SXin Li 
266*67e74705SXin Li   struct B { enum class E; };
267*67e74705SXin Li   template<typename T> enum class B::E { e }; // expected-error {{enumeration cannot be a template}}
268*67e74705SXin Li }
269*67e74705SXin Li 
270*67e74705SXin Li namespace PR16900 {
271*67e74705SXin Li   enum class A;
f(A a)272*67e74705SXin Li   A f(A a) { return -a; } // expected-error {{invalid argument type 'PR16900::A' to unary expression}}
273*67e74705SXin Li }
274*67e74705SXin Li 
275*67e74705SXin Li namespace PR18551 {
276*67e74705SXin Li   enum class A { A };
f()277*67e74705SXin Li   bool f() { return !A::A; } // expected-error {{invalid argument type 'PR18551::A' to unary expression}}
278*67e74705SXin Li }
279*67e74705SXin Li 
280*67e74705SXin Li namespace rdar15124329 {
281*67e74705SXin Li   enum class B : bool { F, T };
282*67e74705SXin Li 
283*67e74705SXin Li   const rdar15124329::B T1 = B::T;
284*67e74705SXin Li   typedef B C;  const C T2 = B::T;
285*67e74705SXin Li 
286*67e74705SXin Li   static_assert(T1 != B::F, "");
287*67e74705SXin Li   static_assert(T2 == B::T, "");
288*67e74705SXin Li }
289*67e74705SXin Li 
290*67e74705SXin Li namespace PR18044 {
291*67e74705SXin Li   enum class E { a };
292*67e74705SXin Li 
293*67e74705SXin Li   int E::e = 0; // expected-error {{does not refer into a class}}
f()294*67e74705SXin Li   void E::f() {} // expected-error {{does not refer into a class}}
295*67e74705SXin Li   struct E::S {}; // expected-error {{no struct named 'S'}}
296*67e74705SXin Li   struct T : E::S {}; // expected-error {{expected class name}}
297*67e74705SXin Li   enum E::E {}; // expected-error {{no enum named 'E'}}
298*67e74705SXin Li   int E::*p; // expected-error {{does not point into a class}}
299*67e74705SXin Li   using E::f; // expected-error {{no member named 'f'}}
300*67e74705SXin Li 
301*67e74705SXin Li   using E::a; // expected-error {{using declaration cannot refer to a scoped enumerator}}
302*67e74705SXin Li   E b = a; // expected-error {{undeclared}}
303*67e74705SXin Li }
304*67e74705SXin Li 
305*67e74705SXin Li namespace test11 {
306*67e74705SXin Li   enum class E { a };
307*67e74705SXin Li   typedef E E2;
f1()308*67e74705SXin Li   E2 f1() { return E::a; }
309*67e74705SXin Li 
f()310*67e74705SXin Li   bool f() { return !f1(); } // expected-error {{invalid argument type 'E2' (aka 'test11::E') to unary expression}}
311*67e74705SXin Li }
312