// RUN: %clang_cc1 -fsyntax-only -verify %s
// RUN: %clang_cc1 -std=c++20 -fsyntax-only -verify %s
// C++ [basic.def.odr]p2:
// An expression is potentially evaluated unless it [...] is the
// operand of the typeid operator and the expression does not
// designate an lvalue of polymorphic class type.
// FIXME: This should really include <typeinfo>, but we don't have that yet.
namespace std {
class type_info;
}
struct Poly {
virtual ~Poly();
};
struct NonPoly { };
template<typename T, typename Result = T>
struct X {
Result f(T t) { return t + t; } // expected-error{{invalid operands}}
void g(T t) {
(void)typeid(f(t)); // expected-note{{here}}
}
};
void test(X<Poly> xp, X<Poly, Poly&> xpr, X<NonPoly> xnp, X<NonPoly, NonPoly&> xnpr) {
// These are okay (although GCC and EDG get them wrong).
xp.g(Poly());
xnp.g(NonPoly());
xnpr.g(NonPoly());
// Triggers an error (as it should);
xpr.g(Poly()); // expected-note{{instantiation of member function}}
}
#if __cplusplus >= 202002L
namespace unevaluated {
struct S {
void f();
};
struct T {
virtual void f();
};
consteval S *null_s() { return nullptr; }
consteval S *make_s() { return new S; }
consteval T *null_t() { return nullptr; }
consteval T *make_t() { return new T; } // #alloc
void func() {
(void)typeid(*null_s());
(void)typeid(*make_s());
(void)typeid(*null_t()); // expected-warning {{expression with side effects will be evaluated despite being used as an operand to 'typeid'}}
(void)typeid(*make_t()); // expected-error {{call to consteval function 'unevaluated::make_t' is not a constant expression}} \
expected-note {{pointer to heap-allocated object is not a constant expression}} \
expected-note@#alloc {{heap allocation performed here}} \
expected-warning {{expression with side effects will be evaluated despite being used as an operand to 'typeid'}}
}
} // namespace unevaluated
#endif