1 // RUN: %clang_cc1 -std=c++2a -Wno-unused-value %s -verify
2 // RUN: %clang_cc1 -std=c++2b -Wno-unused-value %s -verify
4 consteval
int id(int i
) { return i
; }
5 constexpr char id(char c
) { return c
; }
8 constexpr int f(T t
) { // expected-note {{declared here}}
9 return t
+ id(t
); // expected-note 2{{'f<int>' is an immediate function because its body contains a call to a consteval function 'id' and that call is not a constant expression}}
14 auto a
= &f
<char>; // ok, f<char> is not an immediate function
15 auto b
= &f
<int>; // expected-error {{cannot take address of immediate function 'f<int>' outside of an immediate invocation}}
17 static_assert(f(3) == 6); // ok
20 constexpr int g(T t
) { // g<int> is not an immediate function
21 return t
+ id(42); // because id(42) is already a constant
24 template <typename T
, typename F
>
25 constexpr bool is_not(T t
, F f
) {
29 consteval
bool is_even(int i
) { return i
% 2 == 0; }
31 static_assert(is_not(5, is_even
));
33 int x
= 0; // expected-note {{declared here}}
36 constexpr T
h(T t
= id(x
)) { // expected-note {{read of non-const variable 'x' is not allowed in a constant expression}} \
37 // expected-note {{'hh<int>' is an immediate function because its body contains a call to a consteval function 'id' and that call is not a constant expression}}
42 constexpr T
hh() { // hh<int> is an immediate function
43 [[maybe_unused
]] auto x
= h
<T
>();
47 int i
= hh
<int>(); // expected-error {{call to immediate function 'examples::hh<int>' is not a constant expression}} \
48 // expected-note {{in call to 'hh<int>()'}}
56 constexpr int k(int) {
65 constexpr int fdupe(T t
) {
73 a
aa(fdupe
<int>((f
<int>(7))));
76 constexpr int foo(T t
); // expected-note {{declared here}}
78 a
bb(f
<int>(foo
<int>(7))); // expected-error{{call to immediate function 'f<int>' is not a constant expression}} \
79 // expected-note{{undefined function 'foo<int>' cannot be used in a constant expression}}
90 namespace forward_declare_constexpr
{
98 constexpr int f(T t
) {
103 namespace forward_declare_consteval
{
104 template <typename T
>
105 constexpr int f(T t
);
108 auto b
= &f
<int>; // expected-error {{immediate function 'f<int>' used before it is defined}} \
109 // expected-note {{in instantiation of function template specialization}}
111 template <typename T
>
112 constexpr int f(T t
) { // expected-note {{'f<int>' defined here}}
113 return id(t
); // expected-note {{'f<int>' is an immediate function because its body contains a call to a consteval function 'id' and that call is not a constant expression}}
117 namespace constructors
{
118 consteval
int f(int) {
122 constexpr S(auto i
) {
126 constexpr void g(auto i
) {
127 [[maybe_unused
]] S s
{i
};
134 namespace aggregate
{
135 consteval
int f(int);
141 constexpr bool test(auto i
) {
145 consteval
int f(int i
) {
150 static_assert(test(42));
155 namespace ConstevalConstructor
{
156 int x
= 0; // expected-note {{declared here}}
160 constexpr int g(auto t
) {
161 S
s(t
); // expected-note {{'g<int>' is an immediate function because its body contains a call to a consteval constructor 'S' and that call is not a constant expression}}
164 int i
= g(x
); // expected-error {{call to immediate function 'ConstevalConstructor::g<int>' is not a constant expression}} \
165 // expected-note {{read of non-const variable 'x' is not allowed in a constant expression}}
170 namespace Aggregate
{
171 consteval
int f(int); // expected-note {{declared here}}
173 int x
= f(42); // expected-note {{undefined function 'f' cannot be used in a constant expression}} \
174 // expected-note {{'immediate<int>' is an immediate function because its body contains a call to a consteval function 'f' and that call is not a constant expression}}
177 constexpr S
immediate(auto) {
181 void test_runtime() {
182 (void)immediate(0); // expected-error {{call to immediate function 'Aggregate::immediate<int>' is not a constant expression}} \
183 // expected-note {{in call to 'immediate<int>(0)'}}
185 consteval
int f(int i
) {
188 consteval
void test() {
189 constexpr S s
= immediate(0);
190 static_assert(s
.x
== 42);
197 void side_effect(); // expected-note {{declared here}}
198 consteval
int f(int x
) {
199 if (!x
) side_effect(); // expected-note {{non-constexpr function 'side_effect' cannot be used in a constant expression}}
204 int x
= f(0); // expected-error {{call to consteval function 'GH63742::f' is not a constant expression}} \
205 // expected-note {{declared here}} \
206 // expected-note {{in call to 'f(0)'}}
209 SS::SS(){} // expected-note {{in the default initializer of 'x'}}
211 consteval
int f2(int x
) {
212 if (!__builtin_is_constant_evaluated()) side_effect();
222 constinit S2 s2
= {};
232 namespace Defaulted
{
233 consteval
int f(int x
);
240 namespace DefaultedUse
{
241 consteval
int f(int x
); // expected-note {{declared here}}
243 int a
= sizeof(f(0)); // Ok
244 int x
= f(0); // expected-note {{undefined function 'f' cannot be used in a constant expression}}
246 SS() = default; // expected-note {{'SS' is an immediate constructor because the default initializer of 'x' contains a call to a consteval function 'f' and that call is not a constant expression}}
250 [[maybe_unused
]] SS s
; // expected-error {{call to immediate function 'DefaultedUse::SS::SS' is not a constant expression}} \
251 // expected-note {{in call to 'SS()'}}
255 namespace UserDefinedConstructors
{
256 consteval
int f(int x
) {
259 extern int NonConst
; // expected-note 2{{declared here}}
261 struct ConstevalCtr
{
264 consteval
ConstevalCtr(int yy
)
269 ConstevalCtr
c2(NonConst
);
270 // expected-error@-1 {{call to consteval function 'UserDefinedConstructors::ConstevalCtr::ConstevalCtr' is not a constant expression}} \
271 // expected-note@-1 {{read of non-const variable 'NonConst' is not allowed in a constant expression}}
273 struct ImmediateEscalating
{
277 constexpr ImmediateEscalating(T yy
) // expected-note {{ImmediateEscalating<int>' is an immediate constructor because the initializer of 'y' contains a call to a consteval function 'f' and that call is not a constant expression}}
281 ImmediateEscalating
c3(1);
282 ImmediateEscalating
c4(NonConst
);
283 // expected-error@-1 {{call to immediate function 'UserDefinedConstructors::ImmediateEscalating::ImmediateEscalating<int>' is not a constant expression}} \
284 // expected-note@-1 {{read of non-const variable 'NonConst' is not allowed in a constant expression}}
287 struct NonEscalating
{
289 int x
= f(this->y
); // expected-error {{call to consteval function 'UserDefinedConstructors::f' is not a constant expression}} \
290 // expected-note {{declared here}} \
291 // expected-note {{use of 'this' pointer is only allowed within the evaluation of a call to a 'constexpr' member function}}
292 constexpr NonEscalating(int yy
) : y(yy
) {} // expected-note {{in the default initializer of 'x'}}
294 NonEscalating s
= {1};
298 namespace AggregateInit
{
300 consteval
int f(int x
) {
309 constexpr S
test(auto) {
317 namespace GlobalAggregateInit
{
319 consteval
int f(int x
) {
325 int j
= f(i
); // expected-error {{call to consteval function 'GlobalAggregateInit::f' is not a constant expression}} \
326 // expected-note {{implicit use of 'this' pointer is only allowed within the evaluation of a call to a 'constexpr' member function}} \
327 // expected-note {{declared here}}
330 S
s(0); // expected-note {{in the default initializer of 'j'}}
335 consteval
int invalid(); // expected-note 2{{declared here}}
336 constexpr int escalating(auto) {
338 // expected-note@-1 {{'escalating<int>' is an immediate function because its body contains a call to a consteval function 'invalid' and that call is not a constant expression}}
339 // expected-note@-2 2{{undefined function 'invalid' cannot be used in a constant expression}}
342 static constexpr int a
= escalating(0); // expected-note 2{{in call to}}
343 // expected-error@-1 {{call to immediate function 'GH65985::escalating<int>' is not a constant expression}}
344 // expected-error@-2 {{constexpr variable 'a' must be initialized by a constant expression}}
351 consteval
int allocate(); // expected-note 2{{declared here}}
353 struct _Vector_base
{
354 int b
= allocate(); // expected-note 2{{undefined function 'allocate' cannot be used in a constant expression}} \
355 // expected-error {{call to consteval function 'GH66324::allocate' is not a constant expression}} \
356 // expected-note {{declared here}}
360 struct vector
: _Vector_base
{
362 // expected-note@-1 {{'vector' is an immediate constructor because its body contains a call to a consteval function 'allocate' and that call is not a constant expression}}
363 : _Vector_base
{} {} // expected-note {{in the default initializer of 'b'}}
367 // expected-error@-1 {{call to immediate function 'GH66324::vector<void>::vector' is not a constant expression}}
368 // expected-note@-2 {{in call to 'vector()'}}