aboutsummaryrefslogtreecommitdiff
path: root/clang/test/SemaCXX/member-enum-declarations.cpp
blob: e08f6e7a3fcd6dbf74784b82edfe8b578b0bc750 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
// RUN: %clang_cc1 -std=c++11 -fsyntax-only %s -verify
// RUN: %clang_cc1 -std=c++14 -fsyntax-only %s -verify
// RUN: %clang_cc1 -std=c++20 -fsyntax-only %s -verify


namespace ScopedEnumerations {

template <typename T>
struct S1 {
  enum class E : T;
};

template <typename T>
enum class S1<T>::E : T {
  S1_X = 0x123
};

static_assert(static_cast<int>(S1<int>::E::S1_X) == 0x123, "");

template <typename T>
struct S2 {
  static constexpr T f(int) { return 0; };
  enum class E : T;
  static constexpr T f(char) { return 1; };
  enum class E : T { X = f(T{}) };
};

static_assert(static_cast<int>(S2<char>::E::X) == 1, "");

template <typename T>
struct S3 {
  enum class E : T;
  enum class E : T { X = 0x7FFFFF00 }; // expected-error {{cannot be narrowed to type 'char'}} expected-warning {{implicit conversion from 'int' to 'char'}}
};
template struct S3<char>; // expected-note {{in instantiation}}

template <typename T>
struct S4 {
  enum class E : T;
  enum class E : T { S4_X = 5 };
};

auto x4 = S4<int>::E::S4_X;

template <typename T>
T f1() {
  enum class E : T { X_F1, Y_F1, Z_F1 };
  return X_F1;  // expected-error {{use of undeclared identifier 'X_F1'}}
}

const int resf1 = f1<int>();

}


namespace UnscopedEnumerations {

template <typename T>
struct S1 {
  enum E : T;
};

template <typename T>
enum S1<T>::E : T {
  S1_X = 0x123
};

static_assert(static_cast<int>(S1<int>::S1_X) == 0x123, "");

template <typename T>
struct S2 {
  static constexpr T f(int) { return 0; };
  enum E : T;
  static constexpr T f(char) { return 1; };
  enum E : T { S2_X = f(T{}) };
};

static_assert(static_cast<int>(S2<char>::E::S2_X) == 1, "");

template <typename T>
struct S3 {
  enum E : T;
  enum E : T { S3_X = 0x7FFFFF00 }; // expected-error {{cannot be narrowed to type 'char'}} expected-warning {{implicit conversion from 'int' to 'char'}}
};
template struct S3<char>; // expected-note {{in instantiation of template class}}

template <typename T>
struct S4 {
  enum E : T;
  enum E : T { S4_X = 5 };
};

auto x4 = S4<int>::S4_X;

template <typename T>
struct S5 {
  enum E : T;
  T S5_X = 5; // expected-note {{previous definition is here}}
  enum E : T { S5_X = 5 }; // expected-error {{redefinition of 'S5_X'}}
};


template <typename T>
T f1() {
  enum E : T { X_F2, Y_F2, Z_F2 };
  return X_F2;
}

const int resf1 = f1<int>();

}