// This test is for the [class.compare.default]p3 added by P2002R0 // RUN: %clang_cc1 -std=c++2a -verify %s namespace std { struct strong_ordering { int n; constexpr operator int() const { return n; } static const strong_ordering less, equal, greater; }; constexpr strong_ordering strong_ordering::less = {-1}; constexpr strong_ordering strong_ordering::equal = {0}; constexpr strong_ordering strong_ordering::greater = {1}; } struct A { friend bool operator==(const A&, const A&) = default; friend bool operator!=(const A&, const A&) = default; friend std::strong_ordering operator<=>(const A&, const A&) = default; friend bool operator<(const A&, const A&) = default; friend bool operator<=(const A&, const A&) = default; friend bool operator>(const A&, const A&) = default; friend bool operator>=(const A&, const A&) = default; }; struct TestA { friend constexpr bool operator==(const A&, const A&) noexcept; friend constexpr bool operator!=(const A&, const A&) noexcept; friend constexpr std::strong_ordering operator<=>(const A&, const A&) noexcept; friend constexpr bool operator<(const A&, const A&); friend constexpr bool operator<=(const A&, const A&); friend constexpr bool operator>(const A&, const A&); friend constexpr bool operator>=(const A&, const A&); }; // Declaration order doesn't matter, even though the secondary operators need // to know whether the primary ones are constexpr. struct ReversedA { friend bool operator>=(const ReversedA&, const ReversedA&) = default; friend bool operator>(const ReversedA&, const ReversedA&) = default; friend bool operator<=(const ReversedA&, const ReversedA&) = default; friend bool operator<(const ReversedA&, const ReversedA&) = default; friend std::strong_ordering operator<=>(const ReversedA&, const ReversedA&) = default; friend bool operator!=(const ReversedA&, const ReversedA&) = default; friend bool operator==(const ReversedA&, const ReversedA&) = default; }; struct TestReversedA { friend constexpr bool operator>=(const ReversedA&, const ReversedA&); friend constexpr bool operator>(const ReversedA&, const ReversedA&); friend constexpr bool operator<=(const ReversedA&, const ReversedA&); friend constexpr bool operator<(const ReversedA&, const ReversedA&); friend constexpr std::strong_ordering operator<=>(const ReversedA&, const ReversedA&) noexcept; friend constexpr bool operator!=(const ReversedA&, const ReversedA&) noexcept; friend constexpr bool operator==(const ReversedA&, const ReversedA&) noexcept; }; struct B { A a; friend bool operator==(const B&, const B&) = default; friend bool operator!=(const B&, const B&) = default; friend std::strong_ordering operator<=>(const B&, const B&) = default; friend bool operator<(const B&, const B&) = default; friend bool operator<=(const B&, const B&) = default; friend bool operator>(const B&, const B&) = default; friend bool operator>=(const B&, const B&) = default; }; struct TestB { friend constexpr bool operator==(const B&, const B&) noexcept; friend constexpr bool operator!=(const B&, const B&) noexcept; friend constexpr std::strong_ordering operator<=>(const B&, const B&); friend constexpr bool operator<(const B&, const B&); friend constexpr bool operator<=(const B&, const B&); friend constexpr bool operator>(const B&, const B&); friend constexpr bool operator>=(const B&, const B&); }; struct C { friend bool operator==(const C&, const C&); // expected-note {{previous}} expected-note 2{{here}} friend bool operator!=(const C&, const C&) = default; // expected-note {{previous}} friend std::strong_ordering operator<=>(const C&, const C&); // expected-note {{previous}} expected-note 2{{here}} friend bool operator<(const C&, const C&) = default; // expected-note {{previous}} friend bool operator<=(const C&, const C&) = default; // expected-note {{previous}} friend bool operator>(const C&, const C&) = default; // expected-note {{previous}} friend bool operator>=(const C&, const C&) = default; // expected-note {{previous}} }; struct TestC { friend constexpr bool operator==(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr bool operator!=(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr std::strong_ordering operator<=>(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr bool operator<(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr bool operator<=(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr bool operator>(const C&, const C&); // expected-error {{non-constexpr}} friend constexpr bool operator>=(const C&, const C&); // expected-error {{non-constexpr}} }; struct D { A a; C c; A b; friend bool operator==(const D&, const D&) = default; // expected-note {{previous}} friend bool operator!=(const D&, const D&) = default; // expected-note {{previous}} friend std::strong_ordering operator<=>(const D&, const D&) = default; // expected-note {{previous}} friend bool operator<(const D&, const D&) = default; // expected-note {{previous}} friend bool operator<=(const D&, const D&) = default; // expected-note {{previous}} friend bool operator>(const D&, const D&) = default; // expected-note {{previous}} friend bool operator>=(const D&, const D&) = default; // expected-note {{previous}} }; struct TestD { friend constexpr bool operator==(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr bool operator!=(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr std::strong_ordering operator<=>(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr bool operator<(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr bool operator<=(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr bool operator>(const D&, const D&); // expected-error {{non-constexpr}} friend constexpr bool operator>=(const D&, const D&); // expected-error {{non-constexpr}} }; struct E { A a; C c; // expected-note 2{{non-constexpr comparison function would be used to compare member 'c'}} A b; friend constexpr bool operator==(const E&, const E&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator!=(const E&, const E&) = default; friend constexpr std::strong_ordering operator<=>(const E&, const E&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator<(const E&, const E&) = default; friend constexpr bool operator<=(const E&, const E&) = default; friend constexpr bool operator>(const E&, const E&) = default; friend constexpr bool operator>=(const E&, const E&) = default; }; struct E2 : A, C { // expected-note 2{{non-constexpr comparison function would be used to compare base class 'C'}} friend constexpr bool operator==(const E2&, const E2&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator!=(const E2&, const E2&) = default; friend constexpr std::strong_ordering operator<=>(const E2&, const E2&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator<(const E2&, const E2&) = default; friend constexpr bool operator<=(const E2&, const E2&) = default; friend constexpr bool operator>(const E2&, const E2&) = default; friend constexpr bool operator>=(const E2&, const E2&) = default; }; struct F { friend bool operator==(const F&, const F&); // expected-note {{here}} friend constexpr bool operator!=(const F&, const F&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend std::strong_ordering operator<=>(const F&, const F&); // expected-note 4{{here}} friend constexpr bool operator<(const F&, const F&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator<=(const F&, const F&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator>(const F&, const F&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} friend constexpr bool operator>=(const F&, const F&) = default; // expected-error {{cannot be declared constexpr because it invokes a non-constexpr comparison function}} }; // No implicit 'constexpr' if it's not the first declaration. // FIXME: This rule creates problems for reordering of declarations; is this // really the right model? struct G; bool operator==(const G&, const G&); bool operator!=(const G&, const G&); std::strong_ordering operator<=>(const G&, const G&); bool operator<(const G&, const G&); bool operator<=(const G&, const G&); bool operator>(const G&, const G&); bool operator>=(const G&, const G&); struct G { friend bool operator==(const G&, const G&) = default; friend bool operator!=(const G&, const G&) = default; friend std::strong_ordering operator<=>(const G&, const G&) = default; friend bool operator<(const G&, const G&) = default; friend bool operator<=(const G&, const G&) = default; friend bool operator>(const G&, const G&) = default; friend bool operator>=(const G&, const G&) = default; }; bool operator==(const G&, const G&); bool operator!=(const G&, const G&); std::strong_ordering operator<=>(const G&, const G&); bool operator<(const G&, const G&); bool operator<=(const G&, const G&); bool operator>(const G&, const G&); bool operator>=(const G&, const G&); namespace PR44721 { template bool operator==(T const &, T const &) { return true; } template bool operator!=(T const &, U const &) { return true; } template int operator<=>(T const &, T const &) { return 0; } struct S { friend bool operator==(const S &, const S &) = default; friend bool operator<=>(const S &, const S &) = default; int x; }; }