Richard Smith | 762bb9d | 2011-10-13 22:29:44 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -fsyntax-only -pedantic -std=c++11 -verify -triple x86_64-apple-darwin %s |
Douglas Gregor | 1274ccd | 2010-10-08 23:50:27 +0000 | [diff] [blame] | 2 | |
| 3 | enum class E1 { |
| 4 | Val1 = 1L |
| 5 | }; |
| 6 | |
| 7 | enum struct E2 { |
| 8 | Val1 = '\0' |
| 9 | }; |
| 10 | |
| 11 | E1 v1 = Val1; // expected-error{{undeclared identifier}} |
| 12 | E1 v2 = E1::Val1; |
| 13 | |
| 14 | static_assert(sizeof(E1) == sizeof(int), "bad size"); |
| 15 | static_assert(sizeof(E1::Val1) == sizeof(int), "bad size"); |
| 16 | static_assert(sizeof(E2) == sizeof(int), "bad size"); |
| 17 | static_assert(sizeof(E2::Val1) == sizeof(int), "bad size"); |
| 18 | |
| 19 | E1 v3 = E2::Val1; // expected-error{{cannot initialize a variable}} |
| 20 | int x1 = E1::Val1; // expected-error{{cannot initialize a variable}} |
| 21 | |
| 22 | enum E3 : char { |
| 23 | Val2 = 1 |
| 24 | }; |
| 25 | |
| 26 | E3 v4 = Val2; |
| 27 | E1 v5 = Val2; // expected-error{{cannot initialize a variable}} |
| 28 | |
| 29 | static_assert(sizeof(E3) == 1, "bad size"); |
| 30 | |
| 31 | int x2 = Val2; |
| 32 | |
| 33 | int a1[Val2]; |
| 34 | int a2[E1::Val1]; // expected-error{{size of array has non-integer type}} |
| 35 | |
| 36 | int* p1 = new int[Val2]; |
Richard Smith | f39aec1 | 2012-02-04 07:07:42 +0000 | [diff] [blame] | 37 | int* p2 = new int[E1::Val1]; // expected-error{{array size expression must have integral or unscoped enumeration type, not 'E1'}} |
Douglas Gregor | 1274ccd | 2010-10-08 23:50:27 +0000 | [diff] [blame] | 38 | |
| 39 | enum class E4 { |
| 40 | e1 = -2147483648, // ok |
| 41 | e2 = 2147483647, // ok |
Richard Smith | 8ef7b20 | 2012-01-18 23:55:52 +0000 | [diff] [blame] | 42 | e3 = 2147483648 // expected-error{{enumerator value evaluates to 2147483648, which cannot be narrowed to type 'int'}} |
Douglas Gregor | 1274ccd | 2010-10-08 23:50:27 +0000 | [diff] [blame] | 43 | }; |
| 44 | |
| 45 | enum class E5 { |
| 46 | e1 = 2147483647, // ok |
| 47 | e2 // expected-error{{2147483648 is not representable in the underlying}} |
| 48 | }; |
| 49 | |
| 50 | enum class E6 : bool { |
| 51 | e1 = false, e2 = true, |
| 52 | e3 // expected-error{{2 is not representable in the underlying}} |
| 53 | }; |
| 54 | |
| 55 | enum E7 : bool { |
| 56 | e1 = false, e2 = true, |
| 57 | e3 // expected-error{{2 is not representable in the underlying}} |
| 58 | }; |
| 59 | |
| 60 | template <class T> |
| 61 | struct X { |
| 62 | enum E : T { |
| 63 | e1, e2, |
| 64 | e3 // expected-error{{2 is not representable in the underlying}} |
| 65 | }; |
| 66 | }; |
| 67 | |
| 68 | X<bool> X2; // expected-note{{in instantiation of template}} |
| 69 | |
| 70 | enum Incomplete1; // expected-error{{C++ forbids forward references}} |
| 71 | |
| 72 | enum Complete1 : int; |
| 73 | Complete1 complete1; |
| 74 | |
| 75 | enum class Complete2; |
| 76 | Complete2 complete2; |
| 77 | |
| 78 | // All the redeclarations below are done twice on purpose. Tests that the type |
| 79 | // of the declaration isn't changed. |
| 80 | |
| 81 | enum class Redeclare2; // expected-note{{previous use is here}} expected-note{{previous use is here}} |
| 82 | enum Redeclare2; // expected-error{{previously declared as scoped}} |
| 83 | enum Redeclare2; // expected-error{{previously declared as scoped}} |
| 84 | |
| 85 | enum Redeclare3 : int; // expected-note{{previous use is here}} expected-note{{previous use is here}} |
| 86 | enum Redeclare3; // expected-error{{previously declared with fixed underlying type}} |
| 87 | enum Redeclare3; // expected-error{{previously declared with fixed underlying type}} |
| 88 | |
| 89 | enum class Redeclare5; |
| 90 | enum class Redeclare5 : int; // ok |
| 91 | |
| 92 | enum Redeclare6 : int; // expected-note{{previous use is here}} expected-note{{previous use is here}} |
| 93 | enum Redeclare6 : short; // expected-error{{redeclared with different underlying type}} |
| 94 | enum Redeclare6 : short; // expected-error{{redeclared with different underlying type}} |
| 95 | |
| 96 | enum class Redeclare7; // expected-note{{previous use is here}} expected-note{{previous use is here}} |
| 97 | enum class Redeclare7 : short; // expected-error{{redeclared with different underlying type}} |
| 98 | enum class Redeclare7 : short; // expected-error{{redeclared with different underlying type}} |
Douglas Gregor | b907560 | 2011-02-22 02:55:24 +0000 | [diff] [blame] | 99 | |
| 100 | enum : long { |
| 101 | long_enum_val = 10000 |
| 102 | }; |
| 103 | |
| 104 | enum : long x; // expected-error{{unnamed enumeration must be a definition}} \ |
| 105 | // expected-warning{{declaration does not declare anything}} |
Douglas Gregor | 90566c0 | 2011-03-01 17:16:20 +0000 | [diff] [blame] | 106 | |
| 107 | void PR9333() { |
| 108 | enum class scoped_enum { yes, no, maybe }; |
| 109 | scoped_enum e = scoped_enum::yes; |
| 110 | if (e == scoped_enum::no) { } |
| 111 | } |
Douglas Gregor | b6adf2c | 2011-05-05 16:13:52 +0000 | [diff] [blame] | 112 | |
| 113 | // <rdar://problem/9366066> |
| 114 | namespace rdar9366066 { |
| 115 | enum class X : unsigned { value }; |
| 116 | |
| 117 | void f(X x) { |
| 118 | x % X::value; // expected-error{{invalid operands to binary expression ('rdar9366066::X' and 'rdar9366066::X')}} |
| 119 | x % 8; // expected-error{{invalid operands to binary expression ('rdar9366066::X' and 'int')}} |
| 120 | } |
| 121 | } |
John McCall | 57c1300 | 2011-07-06 05:58:41 +0000 | [diff] [blame] | 122 | |
John McCall | 9dc71d2 | 2011-07-06 06:57:57 +0000 | [diff] [blame] | 123 | // Part 1 of PR10264 |
John McCall | 57c1300 | 2011-07-06 05:58:41 +0000 | [diff] [blame] | 124 | namespace test5 { |
| 125 | namespace ns { |
| 126 | typedef unsigned Atype; |
| 127 | enum A : Atype; |
| 128 | } |
| 129 | enum ns::A : ns::Atype { |
| 130 | x, y, z |
| 131 | }; |
| 132 | } |
John McCall | 9dc71d2 | 2011-07-06 06:57:57 +0000 | [diff] [blame] | 133 | |
| 134 | // Part 2 of PR10264 |
| 135 | namespace test6 { |
| 136 | enum A : unsigned; |
| 137 | struct A::a; // expected-error {{incomplete type 'test6::A' named in nested name specifier}} |
| 138 | enum A::b; // expected-error {{incomplete type 'test6::A' named in nested name specifier}} |
| 139 | int A::c; // expected-error {{incomplete type 'test6::A' named in nested name specifier}} |
| 140 | void A::d(); // expected-error {{incomplete type 'test6::A' named in nested name specifier}} |
| 141 | void test() { |
| 142 | (void) A::e; // expected-error {{incomplete type 'test6::A' named in nested name specifier}} |
| 143 | } |
| 144 | } |
Eli Friedman | 19efa3e | 2011-12-06 00:10:34 +0000 | [diff] [blame] | 145 | |
| 146 | namespace PR11484 { |
| 147 | const int val = 104; |
| 148 | enum class test1 { owner_dead = val, }; |
| 149 | } |
Richard Smith | bdad7a2 | 2012-01-10 01:33:14 +0000 | [diff] [blame] | 150 | |
| 151 | namespace N2764 { |
| 152 | enum class E { a, b }; |
| 153 | enum E x1 = E::a; // ok |
| 154 | enum class E x2 = E::a; // expected-error {{reference to scoped enumeration must use 'enum' not 'enum class'}} |
| 155 | |
| 156 | enum F { a, b }; |
| 157 | enum F y1 = a; // ok |
| 158 | enum class F y2 = a; // expected-error {{reference to enumeration must use 'enum' not 'enum class'}} |
| 159 | |
| 160 | struct S { |
| 161 | friend enum class E; // expected-error {{reference to scoped enumeration must use 'enum' not 'enum class'}} |
| 162 | friend enum class F; // expected-error {{reference to enumeration must use 'enum' not 'enum class'}} |
| 163 | |
| 164 | friend enum G {}; // expected-error {{forward reference}} expected-error {{cannot define a type in a friend declaration}} |
| 165 | friend enum class H {}; // expected-error {{cannot define a type in a friend declaration}} |
| 166 | |
| 167 | enum A : int; |
| 168 | A a; |
| 169 | } s; |
| 170 | |
| 171 | enum S::A : int {}; |
| 172 | |
| 173 | enum class B; |
| 174 | } |
| 175 | |
| 176 | enum class N2764::B {}; |
Douglas Gregor | 5bc37f6 | 2012-03-08 02:08:05 +0000 | [diff] [blame] | 177 | |
| 178 | namespace PR12106 { |
| 179 | template<typename E> struct Enum { |
| 180 | Enum() : m_e(E::Last) {} |
| 181 | E m_e; |
| 182 | }; |
| 183 | |
| 184 | enum eCOLORS { Last }; |
| 185 | Enum<eCOLORS> e; |
| 186 | } |
Nick Lewycky | 1659c37 | 2012-03-10 07:47:07 +0000 | [diff] [blame] | 187 | |
| 188 | namespace test7 { |
| 189 | enum class E { e = (struct S*)0 == (struct S*)0 }; |
| 190 | S *p; |
| 191 | } |
Richard Smith | 3343fad | 2012-03-23 23:09:08 +0000 | [diff] [blame] | 192 | |
| 193 | namespace test8 { |
| 194 | template<typename T> struct S { |
| 195 | enum A : int; // expected-note {{here}} |
| 196 | enum class B; // expected-note {{here}} |
| 197 | enum class C : int; // expected-note {{here}} |
Richard Smith | 4ca93d9 | 2012-03-26 04:08:46 +0000 | [diff] [blame] | 198 | enum class D : int; // expected-note {{here}} |
Richard Smith | 3343fad | 2012-03-23 23:09:08 +0000 | [diff] [blame] | 199 | }; |
| 200 | template<typename T> enum S<T>::A { a }; // expected-error {{previously declared with fixed underlying type}} |
| 201 | template<typename T> enum class S<T>::B : char { b }; // expected-error {{redeclared with different underlying}} |
| 202 | template<typename T> enum S<T>::C : int { c }; // expected-error {{previously declared as scoped}} |
Richard Smith | 4ca93d9 | 2012-03-26 04:08:46 +0000 | [diff] [blame] | 203 | template<typename T> enum class S<T>::D : char { d }; // expected-error {{redeclared with different underlying}} |
| 204 | } |
| 205 | |
| 206 | namespace test9 { |
| 207 | template<typename T> struct S { |
| 208 | enum class ET : T; // expected-note 2{{here}} |
| 209 | enum class Eint : int; // expected-note 2{{here}} |
| 210 | }; |
| 211 | template<> enum class S<int>::ET : int {}; |
| 212 | template<> enum class S<char>::ET : short {}; // expected-error {{different underlying type}} |
| 213 | template<> enum class S<int>::Eint : short {}; // expected-error {{different underlying type}} |
| 214 | template<> enum class S<char>::Eint : int {}; |
| 215 | |
| 216 | template<typename T> enum class S<T>::ET : int {}; // expected-error {{different underlying type 'int' (was 'short')}} |
| 217 | template<typename T> enum class S<T>::Eint : T {}; // expected-error {{different underlying type 'short' (was 'int')}} |
| 218 | |
| 219 | // The implicit instantiation of S<short> causes the implicit instantiation of |
| 220 | // all declarations of member enumerations, so is ill-formed, even though we |
| 221 | // never instantiate the definitions of S<short>::ET nor S<short>::Eint. |
| 222 | S<short> s; // expected-note {{in instantiation of}} |
Richard Smith | 3343fad | 2012-03-23 23:09:08 +0000 | [diff] [blame] | 223 | } |
Richard Smith | 38f0df3 | 2012-03-26 04:58:10 +0000 | [diff] [blame] | 224 | |
| 225 | namespace test10 { |
| 226 | template<typename T> int f() { |
| 227 | enum E : int; |
| 228 | enum E : T; // expected-note {{here}} |
| 229 | E x; |
| 230 | enum E : int { e }; // expected-error {{different underlying}} |
| 231 | x = e; |
| 232 | return x; |
| 233 | } |
| 234 | int k = f<int>(); |
| 235 | int l = f<short>(); // expected-note {{here}} |
| 236 | |
| 237 | template<typename T> int g() { |
| 238 | enum class E : int; |
| 239 | enum class E : T; // expected-note {{here}} |
| 240 | E x; |
| 241 | enum class E : int { e }; // expected-error {{different underlying}} |
| 242 | x = E::e; |
| 243 | return (int)x; |
| 244 | } |
| 245 | int m = g<int>(); |
| 246 | int n = g<short>(); // expected-note {{here}} |
| 247 | } |