https://gcc.gnu.org/bugzilla/show_bug.cgi?id=117317
Bug ID: 117317 Summary: An internal compiler error on call to virtual constexpr function via comparison operator in CRTP with multiple inheritance Product: gcc Version: unknown Status: UNCONFIRMED Severity: normal Priority: P3 Component: c++ Assignee: unassigned at gcc dot gnu.org Reporter: kirshamir at gmail dot com Target Milestone: --- The following code results with internal compiler error: class Base { public: constexpr bool operator==(const Base& b) const { return getIdentity() == b.getIdentity(); } private: constexpr virtual int getIdentity() const = 0; }; template<typename ActualType, typename Base1, typename... AdditionalBases> class Comparable: public Base1, public AdditionalBases... { public: constexpr bool operator==(const Comparable&) const = default; private: constexpr int getIdentity() const override { return 1; } }; class A: public Comparable<A, Base> {}; class B: public Comparable<B, Base> {}; class AB: public Comparable<AB, A, B> {}; int main() { constexpr AB ab_1; constexpr AB ab_2; // internal compiler error :-( static_assert(ab_1 == ab_2); } ===== Error: <source>: In instantiation of 'constexpr int Comparable<ActualType, Base1, AdditionalBases>::getIdentity() const [with ActualType = AB; Base1 = A; AdditionalBases = {B}]': <source>:4:27: required from here 4 | return getIdentity() == b.getIdentity(); | ~~~~~~~~~~~^~ <source>:31:24: in 'constexpr' expansion of 'ab_1.AB::Comparable<AB, A, B>.Comparable<AB, A, B>::operator==(ab_2.AB::Comparable<AB, A, B>)' <source>:13:20: in 'constexpr' expansion of '((const Comparable<AB, A, B>*)this)->Comparable<AB, A, B>::A.A::Comparable<A, Base>.Comparable<A, Base>::operator==(<anonymous>.Comparable<AB, A, B>::A.A::Comparable<A, Base>)' <source>:13:20: in 'constexpr' expansion of '((const Comparable<A, Base>*)this)->Comparable<A, Base>::Base.Base::operator==(<anonymous>.Comparable<A, Base>::Base)' <source>:17:5: internal compiler error: in use_thunk, at cp/method.cc:324 17 | } | ^ 0x287e175 diagnostic_context::diagnostic_impl(rich_location*, diagnostic_metadata const*, diagnostic_option_id, char const*, __va_list_tag (*) [1], diagnostic_t) ???:0 0x28945d5 internal_error(char const*, ...) ???:0 0xa90ace fancy_abort(char const*, int, char const*) ???:0 0xd1eb6b emit_associated_thunks(tree_node*) ???:0 0xd1ed0e expand_or_defer_fn(tree_node*) ???:0 0xcc60cd instantiate_decl(tree_node*, bool, bool) ???:0 0xb08a50 maybe_constant_value(tree_node*, tree_node*, mce_value) ???:0 0xd23df0 finish_static_assert(tree_node*, tree_node*, unsigned int, bool, bool) ???:0 0xc9e57a c_parse_file() ???:0 0xdf7f99 c_common_parse_file() ???:0 -------------- Notes: 1. Code compiles with clang: https://godbolt.org/z/j8oMxar8o 2. Removing the constexpr or the virtual from getIdentity() - code compiles: https://godbolt.org/z/G5nMnWMd9 and https://godbolt.org/z/f3PWez1sb 3. Without CRTP code compiles: https://godbolt.org/z/56ah81n6h