Document Number: P1614R0
Date: 2019-03-11
Audience: LWG
Reply-To: Barry Revzin, barry dot revzin at gmail dot com

The Mothership Has Landed
Adding <=> to the Library

Contents

  1. Introduction
  2. Known behavioral changes
    1. Was well-formed, now ill-formed
    2. Was ill-formed, now well-formed
  3. Acknowledgments
  4. Wording
    1. Clause 15: Library Introduction
    2. Clause 16: Language support library
    3. Clause 17: Concepts Library
    4. Clause 18: Diagnostics Library
    5. Clause 19: General utilities library
    6. Clause 20: Strings library
    7. Clause 21: Containers library
    8. Clause 22: Iterators library
    9. Clause 23: Ranges library
    10. Clause 24: Algorithms library
    11. Clause 25: Numeric library
    12. Clause 26: Time library
    13. Clause 27: Localization library
    14. Clause 28: Input/output library
    15. Clause 29: Regular expressions library
    16. Clause 30: Atomic operations library
    17. Clause 31: Thread support library
  5. References

1. Introduction

The work of integrating operator<=> into the library has been performed by multiple different papers, each addressing a different aspect of the integration. In the interest of streamlining review by the Library Working Group, the wording has been combined into a single paper. This is that paper.

In San Diego and Kona, several papers were approved by LEWG adding functionality to the library related to comparisons. What follows is the list of those papers, in alphabetical order, with a brief description of what those papers are. The complete motivation and design rationale for each can be found within the papers themselves.

LEWG's unanimous preference was that operator<=>s be declared as hidden friends.

This initial draft is complete up through Clause 25 and includes coroutine_handle<>. No work has yet been done on Clauses 26-31.

2. Known behavioral changes

There are a few things that will change behavior as a result of all these papers and the chosen direction for declaring operators as hidden friends.

2.1. Was well-formed, now ill-formed

For the preexisting non-member, non-template comparison operators, any comparison that relies on finding the operator in std with regular unqualified lookup will fail:

using namespace std;
struct X { operator error_code() const; };
X{} == X{};          // ok in C++17, ill-formed with this change
X{} == error_code{}; // ok

Here is a more subtle example, reproduced from the LLVM codebase:

struct StringRef {
    StringRef(std::string const&); // NB: non-explicit
    operator std::string() const;  // NB: non-explicit
};
bool operator==(StringRef, StringRef);

bool f(StringRef a, std::string b) {
    return a == b; // (*)
}

In C++17, the marked line is well-formed. The operator== for basic_string is a non-member function template, and so would not be considered a candidate; the only viable candidate is the operator== taking two StringRefs. With the proposed changes, the operator== for basic_string becomes a non-member hidden friend, non-template, which makes it a candidate (converting a to a string). That candidate is ambiguous with the operator==(StringRef, StringRef) candidate - each requires a conversion in one argument, so the call becomes ill-formed.

2.2. Was ill-formed, now well-formed

bool is42(std::variant<int, std::string> const& v) {
    return v == 42; // (*)
}

In C++17, the operator== for variant is a non-member function template and is thus not a viable candidate for the marked line. That check is ill-formed. With the proposed changes, the operator== for variant becomes a non-member hidden friend, non-template, which makes it a candidate (converting 42 to a variant<int, string>). This is arguably a fix, since both variant<int, string> v = 42; and v = 42; are already well-formed, so it is surely reasonable that v == 42 is as well.

3. Acknowledgments

Thank you to Casey Carter for the tremendous wording review.

4. Wording

4.1. Clause 15: Library Introduction

Change 15.4.2.1/2 [expos.only.func]:

The following function is are defined for exposition only to aid in the specification of the library:

and append:

constexpr auto synth-3way =
  []<class T, class U>(const T& t, const U& u)
    requires requires {
      { t < u } -> bool;
      { u < t } -> bool;
    }
  {
    if constexpr (ThreeWayComparableWith<T, U>) {
      return t <=> u;
    } else {
      if (t < u) return weak_ordering::less;
      if (u < t) return weak_ordering::greater;
      return weak_ordering::equivalent;
    }
  };

template<class T, class U=T>
using synth-3way-result = decltype(synth-3way(declval<T&>(), declval<U&>()));

Remove 15.4.2.3 [operators], which begins:

In this library, whenever a declaration is provided for an operator!=, operator>, operator<=, or operator>= for a type T, its requirements and semantics are as follows, unless explicitly specified otherwise.

Add a clause to 15.5.5 [conforming], probably after 15.5.5.4 [global.functions]. Not strictly related to <=> as a whole, but it's a requirement that's currently missing and needs to be added somewhere. See also P1601.

15.5.5.x Hidden friend functions [conforming.hidden.friend]

An implementation shall not provide any additional out-of-class declarations or redeclarations for any non-member function specified as a non-member friend and defined within the body of a class. [ Note: The intent is that such functions are to be found via argument-dependent lookup only. -end note ]

4.2. Clause 16: Language support library

Added: compare_three_way_result, concepts ThreeWayComparable and ThreeWayComparableWith, compare_three_way and compare_XXX_order_fallback

Changed operators for: type_info

Respecified: strong_order(), weak_order(), and partial_order()

Removed: compare_3way(), strong_equal(), and weak_equal()

In 16.7.2 [type.info], remove operator!=:

namespace std {
  class type_info {
  public:
    virtual ~type_info();
    bool operator==(const type_info& rhs) const noexcept;
    bool operator!=(const type_info& rhs) const noexcept;
    bool before(const type_info& rhs) const noexcept;
    size_t hash_code() const noexcept;
    const char* name() const noexcept;
    type_info(const type_info& rhs) = delete; // cannot be copied
    type_info& operator=(const type_info& rhs) = delete; // cannot be copied
  };
}

and

bool operator==(const type_info& rhs) const noexcept;
Effects: Compares the current object with rhs.
Returns: true if the two values describe the same type.
bool operator!=(const type_info& rhs) const noexcept;
Returns: !(*this == rhs).

Add into 16.11.1 [compare.syn]:

namespace std {
  // [cmp.categories], comparison category types
  class weak_equality;
  class strong_equality;
  class partial_ordering;
  class weak_ordering;
  class strong_ordering;

  // named comparison functions
  constexpr bool is_eq  (weak_equality cmp) noexcept    { return cmp == 0; }
  constexpr bool is_neq (weak_equality cmp) noexcept    { return cmp != 0; }
  constexpr bool is_lt  (partial_ordering cmp) noexcept { return cmp < 0; }
  constexpr bool is_lteq(partial_ordering cmp) noexcept { return cmp <= 0; }
  constexpr bool is_gt  (partial_ordering cmp) noexcept { return cmp > 0; }
  constexpr bool is_gteq(partial_ordering cmp) noexcept { return cmp >= 0; }

  // common_type specializations
  template<> struct common_type<strong_equality, partial_ordering>
    { using type = weak_equality; };
  template<> struct common_type<partial_ordering, strong_equality>
    { using type = weak_equality; };
  template<> struct common_type<strong_equality, weak_ordering>
    { using type = weak_equality; };
  template<> struct common_type<weak_ordering, strong_equality>
    { using type = weak_equality; };

  // [cmp.common], common comparison category type  
  template<class... Ts>
  struct common_comparison_category {
    using type = see below;
  };
  template<class... Ts>
    using common_comparison_category_t = typename common_comparison_category<Ts...>::type;  

  // [cmp.concept], concept ThreeWayComparable
  template<class T, class Cat = partial_ordering>
    concept ThreeWayComparable = see below;
  template<class T, class U, class Cat = partial_ordering>
    concept ThreeWayComparableWith = see below;

  // [cmp.result], spaceship invocation result
  template<class T, class U = T> struct compare_three_way_result;

  template<class T, class U = T>
    using compare_three_way_result_t = typename compare_three_way_result<T, U>::type;

  // [cmp.object], spaceship object
  struct compare_three_way;

  // [cmp.alg], comparison algorithms
  template<class T> constexpr strong_ordering strong_order(const T& a, const T& b);
  template<class T> constexpr weak_ordering weak_order(const T& a, const T& b);
  template<class T> constexpr partial_ordering partial_order(const T& a, const T& b);
  template<class T> constexpr strong_equality strong_equal(const T& a, const T& b);
  template<class T> constexpr weak_equality weak_equal(const T& a, const T& b);
  inline namespace unspecified {
    inline constexpr unspecified strong_order = unspecified;
    inline constexpr unspecified weak_order = unspecified;
    inline constexpr unspecified partial_order = unspecified;
    inline constexpr unspecified compare_strong_order_fallback = unspecified;
    inline constexpr unspecified compare_weak_order_fallback = unspecified;
    inline constexpr unspecified compare_partial_order_fallback = unspecified;
  }
}

Change 16.11.2.2 [cmp.weakeq]:

namespace std {
  class weak_equality {
    int value;  // exposition only
    [...]

    // comparisons
    friend constexpr bool operator==(weak_equality v, unspecified) noexcept;
     { return v.value == 0; }
    friend constexpr bool operator!=(weak_equality v, unspecified) noexcept;
    friend constexpr bool operator==(unspecified, weak_equality v) noexcept;
    friend constexpr bool operator!=(unspecified, weak_equality v) noexcept;
    friend constexpr bool operator==(weak_equality v, weak_equality w) noexcept = default;
    friend constexpr weak_equality operator<=>(weak_equality v, unspecified) noexcept;
     { return v; }
    friend constexpr weak_equality operator<=>(unspecified, weak_equality v) noexcept;
     { return v; }
  };

  // valid values' definitions
  inline constexpr weak_equality weak_equality::equivalent(eq::equivalent);
  inline constexpr weak_equality weak_equality::nonequivalent(eq::nonequivalent);
}

Remove the rest of the clause (now defined inline):

constexpr bool operator==(weak_equality v, unspecified) noexcept;
constexpr bool operator==(unspecified, weak_equality v) noexcept;
Returns: v.value == 0.
constexpr bool operator!=(weak_equality v, unspecified) noexcept;
constexpr bool operator!=(unspecified, weak_equality v) noexcept;
Returns: v.value != 0.
constexpr weak_equality operator<=>(weak_equality v, unspecified) noexcept;
constexpr weak_equality operator<=>(unspecified, weak_equality v) noexcept;
Returns: v.

Change 16.11.2.3 [cmp.strongeq]:

namespace std {
  class strong_equality {
    int value;  // exposition only
    [...]

    // comparisons
    friend constexpr bool operator==(strong_equality v, unspecified) noexcept;
      { return v.value == 0; }
    friend constexpr bool operator!=(strong_equality v, unspecified) noexcept;
    friend constexpr bool operator==(unspecified, strong_equality v) noexcept;
    friend constexpr bool operator!=(unspecified, strong_equality v) noexcept;
    friend constexpr bool operator==(strong_equality v, strong_equality w) noexcept = default;
    friend constexpr strong_equality operator<=>(strong_equality v, unspecified) noexcept;
      { return v; }
    friend constexpr strong_equality operator<=>(unspecified, strong_equality v) noexcept;
      { return v; }
  };

  // valid values' definitions
  inline constexpr strong_equality strong_equality::equal(eq::equal);
  inline constexpr strong_equality strong_equality::nonequal(eq::nonequal);
  inline constexpr strong_equality strong_equality::equivalent(eq::equivalent);
  inline constexpr strong_equality strong_equality::nonequivalent(eq::nonequivalent);
}

Remove most of the rest of the clause:

constexpr operator weak_equality() const noexcept;
Returns: value == 0 ? weak_equality::equivalent : weak_equality::nonequivalent.
constexpr bool operator==(strong_equality v, unspecified) noexcept;
constexpr bool operator==(unspecified, strong_equality v) noexcept;
Returns: v.value == 0.
constexpr bool operator!=(strong_equality v, unspecified) noexcept;
constexpr bool operator!=(unspecified, strong_equality v) noexcept;
Returns: v.value != 0.
constexpr strong_equality operator<=>(strong_equality v, unspecified) noexcept;
constexpr strong_equality operator<=>(unspecified, strong_equality v) noexcept;
Returns: v.

Change 16.11.2.4 [cmp.partialord]:

namespace std {
  class partial_ordering {
    int value;          // exposition only
    bool is_ordered;    // exposition only

    [...]
    // conversion
    constexpr operator weak_equality() const noexcept;

    // comparisons
    friend constexpr bool operator==(partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator!=(partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(partial_ordering v, partial_ordering w) noexcept = default;
    friend constexpr bool operator< (partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator> (partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator<=(partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator>=(partial_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(unspecified, partial_ordering v) noexcept;
    friend constexpr bool operator!=(unspecified, partial_ordering v) noexcept;
    friend constexpr bool operator< (unspecified, partial_ordering v) noexcept;
    friend constexpr bool operator> (unspecified, partial_ordering v) noexcept;
    friend constexpr bool operator<=(unspecified, partial_ordering v) noexcept;
    friend constexpr bool operator>=(unspecified, partial_ordering v) noexcept;
    friend constexpr partial_ordering operator<=>(partial_ordering v, unspecified) noexcept;
    friend constexpr partial_ordering operator<=>(unspecified, partial_ordering v) noexcept;
  };

  [...]
}

Remove just the extra == and != operators in 16.11.2.4 [cmp.partialord]/3 and 4:

constexpr bool operator==(partial_ordering v, unspecified) noexcept;
constexpr bool operator< (partial_ordering v, unspecified) noexcept;
constexpr bool operator> (partial_ordering v, unspecified) noexcept;
constexpr bool operator<=(partial_ordering v, unspecified) noexcept;
constexpr bool operator>=(partial_ordering v, unspecified) noexcept;
Returns: For operator@, v.is_ordered && v.value @ 0.
constexpr bool operator==(unspecified, partial_ordering v) noexcept;
constexpr bool operator< (unspecified, partial_ordering v) noexcept;
constexpr bool operator> (unspecified, partial_ordering v) noexcept;
constexpr bool operator<=(unspecified, partial_ordering v) noexcept;
constexpr bool operator>=(unspecified, partial_ordering v) noexcept;
Returns: For operator@, v.is_ordered && 0 @ v.value.
constexpr bool operator!=(partial_ordering v, unspecified) noexcept;
constexpr bool operator!=(unspecified, partial_ordering v) noexcept;
Returns: For operator@, !v.is_ordered || v.value != 0.

Change 16.11.2.5 [cmp.weakord]:

namespace std {
  class weak_ordering {
    int value;  // exposition only

    [...]
    // comparisons
    friend constexpr bool operator==(weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(weak_ordering v, weak_ordering w) noexcept = default;
    friend constexpr bool operator!=(weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator< (weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator> (weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator<=(weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator>=(weak_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(unspecified, weak_ordering v) noexcept;
    friend constexpr bool operator!=(unspecified, weak_ordering v) noexcept;
    friend constexpr bool operator< (unspecified, weak_ordering v) noexcept;
    friend constexpr bool operator> (unspecified, weak_ordering v) noexcept;
    friend constexpr bool operator<=(unspecified, weak_ordering v) noexcept;
    friend constexpr bool operator>=(unspecified, weak_ordering v) noexcept;
    friend constexpr weak_ordering operator<=>(weak_ordering v, unspecified) noexcept;
    friend constexpr weak_ordering operator<=>(unspecified, weak_ordering v) noexcept;
  };

  [...]
};

Remove just the extra == and != operators from 16.11.2.5 [cmp.weakord]/4 and /5:

constexpr bool operator==(weak_ordering v, unspecified) noexcept;
constexpr bool operator!=(weak_ordering v, unspecified) noexcept;
constexpr bool operator< (weak_ordering v, unspecified) noexcept;
constexpr bool operator> (weak_ordering v, unspecified) noexcept;
constexpr bool operator<=(weak_ordering v, unspecified) noexcept;
constexpr bool operator>=(weak_ordering v, unspecified) noexcept;
Returns: v.value @ 0 for operator@.
constexpr bool operator==(unspecified, weak_ordering v) noexcept;
constexpr bool operator!=(unspecified, weak_ordering v) noexcept;
constexpr bool operator< (unspecified, weak_ordering v) noexcept;
constexpr bool operator> (unspecified, weak_ordering v) noexcept;
constexpr bool operator<=(unspecified, weak_ordering v) noexcept;
constexpr bool operator>=(unspecified, weak_ordering v) noexcept;
Returns: 0 @ v.value for operator@.

Change 16.11.2.6 [cmp.strongord]:

namespace std {
  class strong_ordering {
    int value;  // exposition only

    [...]

    // comparisons
    friend constexpr bool operator==(strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(strong_ordering v, strong_ordering w) noexcept = default;
    friend constexpr bool operator!=(strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator< (strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator> (strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator<=(strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator>=(strong_ordering v, unspecified) noexcept;
    friend constexpr bool operator==(unspecified, strong_ordering v) noexcept;
    friend constexpr bool operator!=(unspecified, strong_ordering v) noexcept;
    friend constexpr bool operator< (unspecified, strong_ordering v) noexcept;
    friend constexpr bool operator> (unspecified, strong_ordering v) noexcept;
    friend constexpr bool operator<=(unspecified, strong_ordering v) noexcept;
    friend constexpr bool operator>=(unspecified, strong_ordering v) noexcept;
    friend constexpr strong_ordering operator<=>(strong_ordering v, unspecified) noexcept;
    friend constexpr strong_ordering operator<=>(unspecified, strong_ordering v) noexcept;
  };

  [...]
}

Remove just the extra == and != operators from 16.11.2.6 [cmp.strongord]/6 and /7:

constexpr bool operator==(strong_ordering v, unspecified) noexcept;
constexpr bool operator!=(strong_ordering v, unspecified) noexcept;
constexpr bool operator< (strong_ordering v, unspecified) noexcept;
constexpr bool operator> (strong_ordering v, unspecified) noexcept;
constexpr bool operator<=(strong_ordering v, unspecified) noexcept;
constexpr bool operator>=(strong_ordering v, unspecified) noexcept;
Returns: v.value @ 0 for operator@.
constexpr bool operator==(unspecified, strong_ordering v) noexcept;
constexpr bool operator!=(unspecified, strong_ordering v) noexcept;
constexpr bool operator< (unspecified, strong_ordering v) noexcept;
constexpr bool operator> (unspecified, strong_ordering v) noexcept;
constexpr bool operator<=(unspecified, strong_ordering v) noexcept;
constexpr bool operator>=(unspecified, strong_ordering v) noexcept;
Returns: 0 @ v.value for operator@.

Add a new subclause [cmp.concept] "concept ThreeWayComparable":

template <typename T, typename Cat>
  concept compares-as = // exposition only
    Same<common_comparison_category_t<T, Cat>, Cat>;

template<class T, class U>
  concept partially-ordered-with = // exposition only
    requires(const remove_reference_t<T>& t,
             const remove_reference_t<U>& u) {
      { t < u } -> Boolean;
      { t > u } -> Boolean;
      { t <= u } -> Boolean;
      { t >= u } -> Boolean;
      { u < t } -> Boolean;
      { u > t } -> Boolean;
      { u <= t } -> Boolean;
      { u >= t } -> Boolean;    
};

Let t and u be lvalues of types const remove_reference_t<T> and const remove_reference_t<U> respectively. partially-ordered-with<T, U> is satisfied only if:

template <typename T, typename Cat = partial_ordering>
  concept ThreeWayComparable =
    weakly-equality-comparable-with<T, T> &&
    (!ConvertibleTo<Cat, partial_ordering> || partially-ordered-with<T, T>) &&
    requires(const remove_reference_t<T>& a,
             const remove_reference_t<T>& b) {
      { a <=> b } -> compares-as<Cat>;
    };

Let a and b be lvalues of type const remove_reference_t<T>. T and Cat model ThreeWayComparable<T, Cat> only if:

template <typename T, typename U,
          typename Cat = partial_ordering>
  concept ThreeWayComparableWith = 
    weakly-equality-comparable-with<T, U> &&
    (!ConvertibleTo<Cat, partial_ordering> || partially-ordered-with<T, U>) &&
    ThreeWayComparable<T, Cat> &&
    ThreeWayComparable<U, Cat> &&
    CommonReference<const remove_reference_t<T>&, const remove_reference_t<U>&> &&
    ThreeWayComparable<
      common_reference_t<const remove_reference_t<T>&, const remove_reference_t<U>&>,
      Cat> &&
    requires(const remove_reference_t<T>& t,
             const remove_reference_t<U>& u) {
      { t <=> u } -> compares-as<Cat>;
      { u <=> t } -> compares-as<Cat>;
    };
Let t and u be lvalues of types const remove_reference_t<T> and const remove_reference_t<U>, respectively. Let C be common_reference_t<const remove_reference_t<T>&, const remove_reference_t<U>&>. T, U, and Cat model ThreeWayComparableWith<T, U, Cat> only if:

Add a new subclause [cmp.result] "spaceship invocation result":

The behavior of a program that adds specializations for the compare_three_way_result template defined in this subclause is undefined.

For the compare_three_way_result type trait applied to the types T and U, let t and u denote lvalues of types const remove_reference_t<T> and const remove_reference_t<U>, respectively. If the expression t <=> u is well-formed, the member typedef-name type denotes the type decltype(t <=> u). Otherwise, there is no member type.

Add a new subclause [cmp.object] "spaceship object":

In this subclause, BUILTIN_PTR_3WAY(T, U) for types T and U is a boolean constant expression. BUILTIN_PTR_3WAY(T, U) is true if and only if <=> in the expression declval<T>() <=> declval<U>() resolves to a built-in operator comparing pointers.

struct compare_three_way {
  template<class T, class U>
    requires ThreeWayComparableWith<T,U> || BUILTIN_PTR_3WAY(T, U)
  constexpr auto operator()(T&& t, U&& u) const;

  using is_transparent = unspecified;
};

Expects: If the expression std::forward<T>(t) <=> std::forward<U>(u) results in a call to a built-in operator <=> comparing pointers of type P, the conversion sequences from both T and U to P are equality-preserving ([concepts.equality]).

Effects:

In addition to being available via inclusion of the <compare> header, the class compare_three_way is available when the header <functional> is included.

Replace the entirety of 16.11.4 [cmp.alg]. This wording relies on the specification-only function 3WAY<R> defined in P1186R1.

template<class T> constexpr strong_ordering strong_order(const T& a, const T& b);
Effects: Compares two values and produces a result of type strong_ordering:

template<class T> constexpr weak_ordering weak_order(const T& a, const T& b);
Effects: Compares two values and produces a result of type weak_ordering:

template<class T> constexpr partial_ordering partial_order(const T& a, const T& b);
Effects: Compares two values and produces a result of type partial_ordering:

template<class T> constexpr strong_equality strong_equal(const T& a, const T& b);
Effects: Compares two values and produces a result of type strong_equality:

template<class T> constexpr weak_equality weak_equal(const T& a, const T& b);
Effects: Compares two values and produces a result of type weak_equality:

The name strong_order denotes a customization point object ([customization.point.object]). The expression strong_order(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to the following:

The name weak_order denotes a customization point object ([customization.point.object]). The expression weak_order(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to the following:

The name partial_order denotes a customization point object ([customization.point.object]). The expression partial_order(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to the following:

The name compare_strong_order_fallback denotes a comparison customization point ([customization.point.object]) object. The expression compare_strong_order_fallback(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to:

The name compare_weak_order_fallback denotes a customization point object ([customization.point.object]). The expression compare_weak_order_fallback(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to:

The name compare_partial_order_fallback denotes a customization point object ([customization.point.object]). The expression compare_partial_order_fallback(E, F) for some subexpressions E and F is expression-equivalent ([defns.expression-equivalent]) to:

Change 16.13.1 [coroutine.syn]:

namespace std {
  [...]
  // 16.13.5 noop coroutine
  noop_coroutine_handle noop_coroutine() noexcept;

  // 16.13.3.6 comparison operators:
  constexpr bool operator==(coroutine_handle<> x, coroutine_handle<> y) noexcept;
  constexpr bool operator!=(coroutine_handle<> x, coroutine_handle<> y) noexcept;
  constexpr bool operator<(coroutine_handle<> x, coroutine_handle<> y) noexcept;
  constexpr bool operator>(coroutine_handle<> x, coroutine_handle<> y) noexcept;
  constexpr bool operator<=(coroutine_handle<> x, coroutine_handle<> y) noexcept;
  constexpr bool operator>=(coroutine_handle<> x, coroutine_handle<> y) noexcept;

  // 16.13.6 trivial awaitables
  [...]
}

Change 16.3.3 [coroutine.handle]:

namespace std {
  template <>
  struct coroutine_handle<void>
  {
    [...]
    // 16.13.3.4 resumption
    void operator()() const;
    void resume() const;
    void destroy() const;

    // comparison operators
    friend constexpr bool operator==(coroutine_handle x, coroutine_handle y) noexcept
    { return x.address() == y.address(); }
    friend constexpr strong_ordering operator<=>(coroutine_handle x, coroutine_handle y) noexcept
    { return compare_three_way()(x.address(), y.address()); }
  private:
    void* ptr; // exposition only
  };
  [...]
}

Remove 16.3.3.6 [coroutine.handle.compare] (as it's now all defined in the header):

constexpr bool operator==(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: x.address() == y.address().
constexpr bool operator!=(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: !(x == y).
<constexpr bool operator<(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: less<>()(x.address(), y.address()).
constexpr bool operator>(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: (y < x).
constexpr bool operator<=(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: !(x > y).
constexpr bool operator>=(coroutine_handle<> x, coroutine_handle<> y) noexcept;
Returns: !(x < y).

4.3. Clause 17: Concepts Library

No changes.

4.4. Clause 18: Diagnostics Library

Changed operators for: error_category, error_code, and error_condition

Change 18.5.1 [system_error.syn]

namespace std {
  [...]
  // [syserr.errcondition.nonmembers], non-member functions
  error_condition make_error_condition(errc e) noexcept;

  // [syserr.compare], comparison functions
  bool operator==(const error_code& lhs, const error_code& rhs) noexcept;
  bool operator==(const error_code& lhs, const error_condition& rhs) noexcept;
  bool operator==(const error_condition& lhs, const error_code& rhs) noexcept;
  bool operator==(const error_condition& lhs, const error_condition& rhs) noexcept;
  bool operator!=(const error_code& lhs, const error_code& rhs) noexcept;
  bool operator!=(const error_code& lhs, const error_condition& rhs) noexcept;
  bool operator!=(const error_condition& lhs, const error_code& rhs) noexcept;
  bool operator!=(const error_condition& lhs, const error_condition& rhs) noexcept;
  bool operator< (const error_code& lhs, const error_code& rhs) noexcept;
  bool operator< (const error_condition& lhs, const error_condition& rhs) noexcept;

  // [syserr.hash], hash support
  [...]
}

Change 18.5.2.1 [syserr.errcat.overview]:

namespace std {
  class error_category {
    [...]
    bool operator==(const error_category& rhs) const noexcept;
    bool operator!=(const error_category& rhs) const noexcept;
    bool operator< (const error_category& rhs) const noexcept;
    strong_ordering operator<=>(const error_category& rhs) const noexcept;
  };
  [...]
}

Change 18.5.2.3 [syserr.errcat.nonvirtuals]:

bool operator==(const error_category& rhs) const noexcept;
Returns: this == &rhs.
bool operator!=(const error_category& rhs) const noexcept;
Returns: !(*this == rhs).
bool operator<(const error_category& rhs) const noexcept;
Returns: less<const error_category*>()(this, &rhs).
[Note: less (19.14.7) provides a total ordering for pointers. —end note]
strong_ordering operator<=>(const error_category& rhs) const noexcept;
Returns: compare_three_way()(this, &rhs).
[Note: compare_three_way (cmp.object) provides a total ordering for pointers. —end note]

Change 18.5.3.1 [syserr.errcode.overview]:

namespace std {
  class error_code {
    [...]
    // [syserr.compare], comparison functions
    friend bool operator==(const error_code&, const error_code&) { see below; }
    friend strong_ordering operator<=>(const error_code&, const error_code&) { see below; }
    friend bool operator==(const error_code&, const error_condition&) { see below; }
  private:
    int val_;                   // exposition only
    const error_category* cat_; // exposition only
  };

  [...]
}

Change 18.5.4.1 [syserr.errcondition.overview]:

namespace std {
  class error_condition {
  public:
    [...]
    // [syserr.compare], comparison functions
    friend bool operator==(const error_condition&, const error_condition&) { see below; }
    friend strong_ordering operator<=>(const error_condition&, const error_condition&) { see below; }
    friend bool operator==(const error_condition&, const error_code&) { see below; }
  private:
    int val_;                   // exposition only
    const error_category* cat_; // exposition only
  };
}

Change 18.5.5 [syserr.compare]

friend bool operator==(const error_code& lhs, const error_code& rhs) noexcept;
Returns: lhs.category() == rhs.category() && lhs.value() == rhs.value()
Remarks: This function is to be found via argument-dependent lookup only.
friend bool operator==(const error_code& lhs, const error_condition& rhs) noexcept;
Returns: lhs.category().equivalent(lhs.value(), rhs) || rhs.category().equivalent(lhs, rhs.value())
Remarks: This function is to be found via argument-dependent lookup only.
friend bool operator==(const error_condition& lhs, const error_code& rhs) noexcept;
Returns: rhs.category().equivalent(rhs.value(), lhs) || lhs.category().equivalent(rhs, lhs.value())
Remarks: This function is to be found via argument-dependent lookup only.
friend bool operator==(const error_condition& lhs, const error_condition& rhs) noexcept;
Returns: lhs.category() == rhs.category() && lhs.value() == rhs.value()
Remarks: This function is to be found via argument-dependent lookup only.
bool operator!=(const error_code& lhs, const error_code& rhs) noexcept;
bool operator!=(const error_code& lhs, const error_condition& rhs) noexcept;
bool operator!=(const error_condition& lhs, const error_code& rhs) noexcept;
bool operator!=(const error_condition& lhs, const error_condition& rhs) noexcept;
Returns: !(lhs == rhs).
bool operator<(const error_code& lhs, const error_code& rhs) noexcept;
Returns: lhs.category() < rhs.category() || (lhs.category() == rhs.category() && lhs.value() < rhs.value())
bool operator<(const error_condition& lhs, const error_condition& rhs) noexcept;
Returns: lhs.category() < rhs.category() || (lhs.category() == rhs.category() && lhs.value() < rhs.value())
friend strong_ordering operator<=>(const error_code& lhs, const error_code& rhs) noexcept;
Effects: Equivalent to:
if (auto c = lhs.category() <=> rhs.category(); c != 0) return c;
return lhs.value() <=> rhs.value();

Remarks: This function is to be found via argument-dependent lookup only.
friend strong_ordering operator<=>(const error_condition& lhs, const error_condition& rhs) noexcept;
Effects: Equivalent to:
if (auto c = lhs.category() <=> rhs.category(); c != 0) return c;
return lhs.value() <=> rhs.value();

Remarks: This function is to be found via argument-dependent lookup only.

4.5. Clause 19: General utilities library

Changed operators for: pair, tuple, optional, variant, monostate, bitset, allocator, unique_ptr, shared_ptr, memory_resource, polymorphic_allocator, scoped_allocator_adaptor, function, type_index

Change 19.2.1 [utility.syn]

#include  // see 16.10.1

namespace std {
  [...]
  // 19.4, class template pair
  template<class T1, class T2>
  struct pair;

  // 19.4.3, pair specialized algorithms
  template<class T1, class T2>
  constexpr bool operator==(const pair<T1, T2>&, const pair<T1, T2>&);
  template<class T1, class T2>
  constexpr bool operator!=(const pair<T1, T2>&, const pair<T1, T2>&);
  template<class T1, class T2>
  constexpr bool operator< (const pair<T1, T2>&, const pair<T1, T2>&);
  template<class T1, class T2>
  constexpr bool operator> (const pair<T1, T2>&, const pair<T1, T2>&);
  template<class T1, class T2>
  constexpr bool operator<=(const pair<T1, T2>&, const pair<T1, T2>&);
  template<class T1, class T2>
  constexpr bool operator>=(const pair<T1, T2>&, const pair<T1, T2>&);

  [...]
}

Change 19.4.2 [pairs.pair]:

namespace std {
template<class T1, class T2>
struct pair {
  [...]
  constexpr void swap(pair& p) noexcept(see below);

  friend constexpr bool operator==(const pair&, const pair&) = default;
  friend constexpr common_comparison_category_t<synth-3way-result<T1>, synth-3way-result<T2>>
    operator<=>(const pair&, const pair&)
    { see below }
};

[...]

constexpr void swap(pair& p) noexcept(see below);
Requires: first shall be swappable with (15.5.3.2) p.first and second shall be swappable with p.second.
Effects: Swaps first with p.first and second with p.second.
Remarks: The expression inside noexcept is equivalent to: is_nothrow_swappable_v<first_type> && is_nothrow_swappable_v<second_type>
friend constexpr common_comparison_category_t<synth-3way-result<T1>, synth-3way-result<T2>>
  operator<=>(const pair& lhs, const pair& rhs);
Effects: Equivalent to:
if (auto c = synth-3way(lhs.first, rhs.first); c != 0) return c;
return synth-3way(lhs.second, rhs.second);
Remarks: This function is to be found via argument-dependent lookup only.

Change 19.4.3 [pairs.spec].

template<class T1, class T2>
constexpr bool operator==(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: x.first == y.first && x.second == y.second.
template<class T1, class T2>
constexpr bool operator!=(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: !(x == y).
template<class T1, class T2>
constexpr bool operator<(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: x.first < y.first || (!(y.first < x.first) && x.second < y.second).
template<class T1, class T2>
constexpr bool operator>(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: y < x.
template<class T1, class T2>
constexpr bool operator<=(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: !(y < x).
template<class T1, class T2>
constexpr bool operator>=(const pair<T1, T2>& x, const pair<T1, T2>& y);
Returns: !(x < y).
template<class T1, class T2>
constexpr void swap(pair<T1, T2>& x, pair<T1, T2>& y) noexcept(noexcept(x.swap(y)));
[...]

Change 19.5.2 [tuple.syn]:

namespace std {
  // 19.5.3, class template tuple
  template<class... Types>
    class tuple;

  [...]  

  template<class T, class... Types>
    constexpr const T&& get(const tuple<Types...>&& t) noexcept;

  // 19.5.3.8, relational operators
  template<class... TTypes, class... UTypes>
    constexpr bool operator==(const tuple<TTypes...>&, const tuple<UTypes...>&);
  template<class... TTypes, class... UTypes>
    constexpr bool operator!=(const tuple<TTypes...>&, const tuple<UTypes...>&);
  template<class... TTypes, class... UTypes>
    constexpr bool operator<(const tuple<TTypes...>&, const tuple<UTypes...>&);
  template<class... TTypes, class... UTypes>
    constexpr bool operator>(const tuple<TTypes...>&, const tuple<UTypes...>&);
  template<class... TTypes, class... UTypes>
    constexpr bool operator<=(const tuple<TTypes...>&, const tuple<UTypes...>&);
  template<class... TTypes, class... UTypes>
    constexpr bool operator>=(const tuple<TTypes...>&, const tuple<UTypes...>&);

  // 19.5.3.9, allocator-related traits
  template<class... Types, class Alloc>
    struct uses_allocator<tuple<Types...>, Alloc>;  

  [...]
}

Change 19.5.3 [tuple.tuple]:

namespace std {
template
class tuple {
public:
  [...]

  // 19.5.3.3, tuple swap
  constexpr void swap(tuple&) noexcept(see below );

  // 19.5.3.8, tuple relational operators
  template<class... UTypes>
    friend constexpr bool operator==(const tuple&, const tuple<UTypes...>&)  
    { see below }
  template<class... UTypes>
    friend constexpr common_comparison_category_t<synth-3way-result<Types, UTypes>...>
      operator<=>(const tuple&, const tuple<UTypes...>&)
    { see below }
};

Change 19.5.3.8 [tuple.rel]:

template<class... TTypes, class... UTypes>
  constexpr bool operator==(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
template<class... UTypes>
  friend constexpr bool operator==(const tuple&, const tuple<UTypes...>&)
Requires: For all i, where 0 <= i and i < sizeof...(TTypes Types), get<i>(t) == get<i>(u) is a well-formed expression returning a type that is convertible to bool. sizeof...(TTypes Types) == sizeof...(UTypes).
Returns: true if get<i>(t) == get<i>(u) for all i, otherwise false. For any two zero-length tuples e and f, e == f returns true.
Effects: The elementary comparisons are performed in order from the zeroth index upwards. No comparisons or element accesses are performed after the first equality comparison that evaluates to false.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... TTypes, class... UTypes>
  constexpr bool operator!=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
Returns: !(t == u).
template<class... TTypes, class... UTypes>
constexpr bool operator<(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
template<class... UTypes>
  friend constexpr common_comparison_category_t<synth-3way-result<Types, UTypes>...>
    operator<=>(const tuple& t, const tuple<UTypes...>& u);
Requires: For all i, where 0 <= i and i < sizeof...(Types), both get<i>(t) < get<i>(u) and get<i>(u) < get<i>(t) are well-formed expressions returning types that are convertible to bool synth-3way(get<i>(t), get<i>(u)) is a well-formed expression. sizeof...(TTypes Types) == sizeof...(UTypes).
Returns: The result of a lexicographical comparison between t and u. The result is defined as: (bool)(get<0>(t) < get<0>(u)) || (!(bool)(get<0>(u) < get<0>(t)) && ttail < utail), where rtail for some tuple r is a tuple containing all but the first element of r. For any two zero-length tuples e and f, e < f returns false.
Effects: Performs a lexicographical comparison between t and u. For any two zero-length tuples t and u, t <=> u returns strong_ordering::equal. Otherwise, equivalent to:
auto c = synth-3way(get<0>(t), get<0>(u));
return (c != 0) ? c : (ttail <=> utail);
where rtail for some tuple r is a tuple containing all but the first element of r.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... TTypes, class... UTypes>
constexpr bool operator>(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
Returns: u < t.
template<class... TTypes, class... UTypes>
constexpr bool operator<=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
Returns: !(u < t)
template<class... TTypes, class... UTypes>
constexpr bool operator>=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);
Returns: !(t < u)
[Note: The above definitions for comparison functions do not require ttail (or utail) to be constructed. It may not even be possible, as t and u are not required to be copy constructible. Also, all comparison functions are short circuited; they do not perform element accesses beyond what is required to determine the result of the comparison. —end note]

Change 19.6.2 [optional.syn]:

namespace std {
  [...]
  // [optional.bad.access], class bad_optional_access
  class bad_optional_access;

  // [optional.relops], relational operators
  template<class T, class U>
    constexpr bool operator==(const optional<T>&, const optional<U>&);
  template<class T, class U>
    constexpr bool operator!=(const optional<T>&, const optional<U>&);
  template<class T, class U>
    constexpr bool operator<(const optional<T>&, const optional<U>&);
  template<class T, class U>
    constexpr bool operator>(const optional<T>&, const optional<U>&);
  template<class T, class U>
    constexpr bool operator<=(const optional<T>&, const optional<U>&);
  template<class T, class U>
    constexpr bool operator>=(const optional<T>&, const optional<U>&);

  // [optional.nullops], comparison with nullopt
  template<class T> constexpr bool operator==(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator==(nullopt_t, const optional<T>&) noexcept;
  template<class T> constexpr bool operator!=(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator!=(nullopt_t, const optional<T>&) noexcept;
  template<class T> constexpr bool operator<(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator<(nullopt_t, const optional<T>&) noexcept;
  template<class T> constexpr bool operator>(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator>(nullopt_t, const optional<T>&) noexcept;
  template<class T> constexpr bool operator<=(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator<=(nullopt_t, const optional<T>&) noexcept;
  template<class T> constexpr bool operator>=(const optional<T>&, nullopt_t) noexcept;
  template<class T> constexpr bool operator>=(nullopt_t, const optional<T>&) noexcept;

  // [optional.comp_with_t], comparison with T
  template<class T, class U> constexpr bool operator==(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator==(const T&, const optional<U>&);
  template<class T, class U> constexpr bool operator!=(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator!=(const T&, const optional<U>&);
  template<class T, class U> constexpr bool operator<(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator<(const T&, const optional<U>&);
  template<class T, class U> constexpr bool operator>(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator>(const T&, const optional<U>&);
  template<class T, class U> constexpr bool operator<=(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator<=(const T&, const optional<U>&);
  template<class T, class U> constexpr bool operator>=(const optional<T>&, const U&);
  template<class T, class U> constexpr bool operator>=(const T&, const optional<U>&);

  // [optional.specalg], specialized algorithms
  template<class T>
    void swap(optional<T>&, optional<T>&) noexcept(see below);
  [...]
}

Change 19.6.3 [optional.optional]:

namespace std {
  template<class T>
  class optional {
  public:
    [...]

    // [optional.mod], modifiers
    void reset() noexcept;

    // [optional.relops], relational operators
    template<class U>
      friend constexpr bool operator==(const optional&, const optional<U>&) { see below }
    template<class U>
      friend constexpr bool operator!=(const optional&, const optional<U>&) { see below }
    template<class U>
      friend constexpr bool operator<(const optional&, const optional<U>&) { see below }
    template<class U>
      friend constexpr bool operator>(const optional&, const optional<U>&) { see below }
    template<class U>
      friend constexpr bool operator<=(const optional&, const optional<U>&) { see below }
    template<class U>
      friend constexpr bool operator>=(const optional&, const optional<U>&) { see below }
    template<ThreeWayComparableWith<T> U>
      friend constexpr compare_three_way_result_t<T,U>
        operator<=>(const optional&, const optional<U>&)
        { see below }

    // comparison with nullopt
    friend constexpr bool operator==(const optional& x, nullopt_t) { return !x; }
    friend constexpr strong_ordering operator<=>(const optional& x, nullopt_t) { return bool(x) <=> false; }

    // [optional.comp_with_t], comparison with T
    template<class U> friend constexpr bool operator==(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator==(const U&, const optional&) { see below }
    template<class U> friend constexpr bool operator!=(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator!=(const U&, const optional&) { see below }
    template<class U> friend constexpr bool operator<(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator<(const U&, const optional&) { see below }
    template<class U> friend constexpr bool operator>(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator>(const U&, const optional&) { see below }
    template<class U> friend constexpr bool operator<=(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator<=(const U&, const optional&) { see below }
    template<class U> friend constexpr bool operator>=(const optional&, const U&) { see below }
    template<class U> friend constexpr bool operator>=(const U&, const optional&) { see below }
    template<ThreeWayComparableWith<T> U>
      friend constexpr compare_three_way_result_t<T,U>
        operator<=>(const optional&, const U&)
        { see below }

  private:
    T *val;         // exposition only
  };
}

Change 19.6.6 [optional.relops]:

template<class T, class U> friend constexpr bool operator==(const optional<T>& x, const optional<U>& y);
RequiresMandates: The expression *x == *y shall be well-formed and its result shall be convertible to bool. [Note: T need not be Cpp17EqualityComparable. —end note]
Returns: If bool(x) != bool(y), false; otherwise if bool(x) == false, true; otherwise *x == *y.
Remarks: Specializations of this function template for which *x == *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator!=(const optional<T>& x, const optional<U>& y);
RequiresMandates: The expression *x != *y shall be well-formed and its result shall be convertible to bool.
Returns: If bool(x) != bool(y), true; otherwise, if bool(x) == false, false; otherwise *x != *y.
Remarks: Specializations of this function template for which *x != *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<(const optional<T>& x, const optional<U>& y);
RequiresMandates: *x < *y shall be well-formed and its result shall be convertible to bool.
Returns: If !y, false; otherwise, if !x, true; otherwise *x < *y.
Remarks: Specializations of this function template for which *x < *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>(const optional<T>& x, const optional<U>& y);
RequiresMandates: The expression *x > *y shall be well-formed and its result shall be convertible to bool.
Returns: If !x, false; otherwise, if !y, true; otherwise *x > *y. Remarks: Specializations of this function template for which *x > *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<=(const optional<T>& x, const optional<U>& y);
RequiresMandates: The expression *x <= *y shall be well-formed and its result shall be convertible to bool.
Returns: If !x, true; otherwise, if !y, false; otherwise *x <= *y.
Remarks: Specializations of this function template for which *x <= *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>=(const optional<T>& x, const optional<U>& y);
RequiresMandates: The expression *x >= *y shall be well-formed and its result shall be convertible to bool.
Returns: If !y, true; otherwise, if !x, false; otherwise *x >= *y.
Remarks: Specializations of this function template for which *x >= *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.
template<ThreeWayComparableWith<T> U>
  friend constexpr compare_three_way_result_t<T,U>
    operator<=>(const optional& x, const optional<U>& y);
Returns: If x && y, *x <=> *y; otherwise bool(x) <=> bool(y).
Remarks: Specializations of this function template for which *x <=> *y is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.

Remove 19.6.7 [optional.nullops] (it is now fully expressed by the two hidden friends defined in the header):

template<class T> constexpr bool operator==(const optional<T>& x, nullopt_t) noexcept;
template<class T> constexpr bool operator==(nullopt_t, const optional<T>& x) noexcept;
Returns: !x.
template<class T> constexpr bool operator!=(const optional<T>& x, nullopt_t) noexcept;
template<class T> constexpr bool operator!=(nullopt_t, const optional<T>& x) noexcept;
Returns: bool(x). [...]

Change 19.6.8 [optional.comp_with_t]:

template<class T, class U> friend constexpr bool operator==(const optional<T>& x, const U& v);
RequiresMandates: The expression *x == v shall be well-formed and its result shall be convertible to bool. [Note: T need not be Cpp17EqualityComparable. —end note]
Effects: Equivalent to: return bool(x) ? *x == v : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator==(const TU& v, const optional<U>& x);
RequiresMandates: The expression v == *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v == *x : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator!=(const optional<T>& x, const U& v);
RequiresMandates: The expression *x != v shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? *x != v : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator!=(const TU& v, const optional<U>& x);
RequiresMandates: The expression v != *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v != *x : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<(const optional<T>& x, const U& v);
RequiresMandates: The expression *x < v shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? *x < v : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<(const TU& v, const optional<U>& x);
RequiresMandates: The expression v < *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v < *x : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>(const optional<T>& x, const U& v);
RequiresMandates: The expression *x > v shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? *x > v : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>(const TU& v, const optional<U>& x);
RequiresMandates: The expression v > *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v > *x : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<=(const optional<T>& x, const U& v);
RequiresMandates: The expression *x <= v shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? *x <= v : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator<=(const TU& v, const optional<U>& x);
RequiresMandates: The expression v <= *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v <= *x : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>=(const optional<T>& x, const U& v);
RequiresMandates: The expression *x >= v shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? *x >= v : false;
Remarks: This function is to be found via argument-dependent lookup only.
template<class T, class U> friend constexpr bool operator>=(const TU& v, const optional<U>& x);
RequiresMandates: The expression v >= *x shall be well-formed and its result shall be convertible to bool.
Effects: Equivalent to: return bool(x) ? v >= *x : true;
Remarks: This function is to be found via argument-dependent lookup only.
template<ThreeWayComparableWith<T> U>
  friend constexpr compare_three_way_result_t<T,U>
    operator<=>(const optional& x, const U& v);
Effects: Equivalent to: return bool(x) ? *x <=> v : strong_ordering::less;
Remarks: This function is to be found via argument-dependent lookup only.

Change 19.7.2 [variant.syn]:

namespace std {
  [...]
  template<class T, class... Types>
    constexpr add_pointer_t<T>
      get_if(variant<Types...>*) noexcept;
  template<class T, class... Types>
    constexpr add_pointer_t<const T>
      get_if(const variant<Types...>*) noexcept;

  // [variant.relops], relational operators
  template<class... Types>
    constexpr bool operator==(const variant<Types...>&, const variant<Types...>&);
  template<class... Types>
    constexpr bool operator!=(const variant<Types...>&, const variant<Types...>&);
  template<class... Types>
    constexpr bool operator<(const variant<Types...>&, const variant<Types...>&);
  template<class... Types>
    constexpr bool operator>(const variant<Types...>&, const variant<Types...>&);
  template<class... Types>
    constexpr bool operator<=(const variant<Types...>&, const variant<Types...>&);
  template<class... Types>
    constexpr bool operator>=(const variant<Types...>&, const variant<Types...>&);

  // [variant.visit], visitation
  [...]  

  // [variant.monostate], class monostate
  struct monostate;

  // [variant.monostate.relops], monostate relational operators
  constexpr bool operator==(monostate, monostate) noexcept;
  constexpr bool operator!=(monostate, monostate) noexcept;
  constexpr bool operator<(monostate, monostate) noexcept;
  constexpr bool operator>(monostate, monostate) noexcept;
  constexpr bool operator<=(monostate, monostate) noexcept;
  constexpr bool operator>=(monostate, monostate) noexcept;

  // [variant.specalg], specialized algorithms
  template<class... Types>
    void swap(variant<Types...>&, variant<Types...>&) noexcept(see below);
  [...]
}

Change 19.7.3 [variant.variant]:

namespace std {
  template<class... Types>
  class variant {
  public:
    [...]

    // [variant.swap], swap
    void swap(variant&) noexcept(see below);

    // [variant.relops], relational operators
    friend constexpr bool operator==(const variant&, const variant&) { see below }
    friend constexpr bool operator!=(const variant&, const variant&) { see below }
    friend constexpr bool operator<(const variant&, const variant&) { see below }
    friend constexpr bool operator>(const variant&, const variant&) { see below }
    friend constexpr bool operator<=(const variant&, const variant&) { see below }
    friend constexpr bool operator>=(const variant&, const variant&) { see below }
    friend constexpr common_comparison_category_t<compare_three_way_result_t<Types>...>
      operator<=>(const variant&, const variant&)
        requires (ThreeWayComparable<Types> && ...)
        { see below }
  };
}

Change 19.7.6 [variant.relops]:

template<class... Types>
friend constexpr bool operator==(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) == get<i>(w) is a valid expression returning a type that is convertible to bool, for all i.
Returns: If v.index() != w.index(), false; otherwise if v.valueless_by_exception(), true; otherwise get<i>(v) == get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>
friend constexpr bool operator!=(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) != get<i>(w) is a valid expression returning a type that is convertible to bool, for all i. Returns: If v.index() != w.index(), true; otherwise if v.valueless_by_exception(), false; otherwise get<i>(v) != get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>
friend constexpr bool operator<(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) < get<i>(w) is a valid expression returning a type that is convertible to bool, for all i.
Returns: If w.valueless_by_exception(), false; otherwise if v.valueless_by_exception(), true; otherwise, if v.index() < w.index(), true; otherwise if v.index() > w.index(), false; otherwise get<i>(v) < get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>
friend constexpr bool operator>(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) > get<i>(w) is a valid expression returning a type that is convertible to bool, for all i.
Returns: If v.valueless_by_exception(), false; otherwise if w.valueless_by_exception(), true; otherwise, if v.index() > w.index(), true; otherwise if v.index() < w.index(), false; otherwise get<i>(v) > get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>
friend constexpr bool operator<=(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) <= get<i>(w) is a valid expression returning a type that is convertible to bool, for all i.
Returns: If v.valueless_by_exception(), true; otherwise if w.valueless_by_exception(), false; otherwise, if v.index() < w.index(), true; otherwise if v.index() > w.index(), false; otherwise get<i>(v) <= get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>
friend constexpr bool operator>=(const variant<Types...>& v, const variant<Types...>& w);
RequiresMandates: get<i>(v) >= get<i>(w) is a valid expression returning a type that is convertible to bool, for all i.
Returns: If w.valueless_by_exception(), true; otherwise if v.valueless_by_exception(), false; otherwise, if v.index() > w.index(), true; otherwise if v.index() < w.index(), false; otherwise get<i>(v) >= get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.
constexpr common_comparison_category_t<compare_three_way_result_t<Types>...>
  friend operator<=>(const variant& v, const variant& w)
    requires (ThreeWayComparable<Types> && ...);
Returns: Let c be (v.index() + 1) <=> (w.index() + 1). If c != 0, c. Otherwise, get<i>(v) <=> get<i>(w) with i being v.index().
Remarks: This function is to be found via argument-dependent lookup only.

Change 19.7.8 [variant.monostate]:

struct monostate{};
struct monostate {
  friend constexpr bool operator==(monostate, monostate) noexcept = default;
  friend constexpr strong_ordering operator<=>(monostate, monostate) noexcept = default;
};
[Note: monostate objects have only a single state; they thus always compare equal. —end note]

Remove 19.7.9 [variant.monostate.relops]:

constexpr bool operator==(monostate, monostate) noexcept { return true; }
constexpr bool operator!=(monostate, monostate) noexcept { return false; }
constexpr bool operator<(monostate, monostate) noexcept { return false; }
constexpr bool operator>(monostate, monostate) noexcept { return false; }
constexpr bool operator<=(monostate, monostate) noexcept { return true; }
constexpr bool operator>=(monostate, monostate) noexcept { return true; }
[Note: monostate objects have only a single state; they thus always compare equal. —end note]

Change 19.9.2 [template.bitset]:

namespace std {
  template<size_t N> class bitset {
  public:
    [...]
    constexpr size_t size() const noexcept;
    bool operator==(const bitset<N>& rhs) const noexcept;
    bool operator!=(const bitset<N>& rhs) const noexcept;
    bool test(size_t pos) const;
    [...]
  };
}

Change 19.9.2.2 [bitset.members]:

bool operator==(const bitset<N>& rhs) const noexcept;
Returns: true if the value of each bit in *this equals the value of the corresponding bit in rhs.
bool operator!=(const bitset<N>& rhs) const noexcept;
Returns: true if !(*this == rhs).

Change 19.10.2 [memory.syn]:

namespace std {
  [...]
  // [default.allocator], the default allocator
  template<class T> class allocator;
  template<class T, class U>
    bool operator==(const allocator<T>&, const allocator<U>&) noexcept;
  template<class T, class U>
    bool operator!=(const allocator<T>&, const allocator<U>&) noexcept;
  [...]
  template<class T, class D> 
    void swap(unique_ptr<T, D>& x, unique_ptr<T, D>& y) noexcept;

  template<class T1, class D1, class T2, class D2>
    bool operator==(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
  template<class T1, class D1, class T2, class D2>
    bool operator!=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
  template<class T1, class D1, class T2, class D2>
    bool operator<(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
  template<class T1, class D1, class T2, class D2>
    bool operator>(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
  template<class T1, class D1, class T2, class D2>
    bool operator<=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
  template<class T1, class D1, class T2, class D2>
    bool operator>=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);

  template<class T, class D>
    bool operator==(const unique_ptr<T, D>& x, nullptr_t) noexcept;
  template<class T, class D>
    bool operator==(nullptr_t, const unique_ptr<T, D>& y) noexcept;
  template<class T, class D>
    bool operator!=(const unique_ptr<T, D>& x, nullptr_t) noexcept;
  template<class T, class D>
    bool operator!=(nullptr_t, const unique_ptr<T, D>& y) noexcept;
  template<class T, class D>
    bool operator<(const unique_ptr<T, D>& x, nullptr_t);
  template<class T, class D>
    bool operator<(nullptr_t, const unique_ptr<T, D>& y);
  template<class T, class D>
    bool operator>(const unique_ptr<T, D>& x, nullptr_t);
  template<class T, class D>
    bool operator>(nullptr_t, const unique_ptr<T, D>& y);
  template<class T, class D>
    bool operator<=(const unique_ptr<T, D>& x, nullptr_t);
  template<class T, class D>
    bool operator<=(nullptr_t, const unique_ptr<T, D>& y);
  template<class T, class D>
    bool operator>=(const unique_ptr<T, D>& x, nullptr_t);
  template<class T, class D>
    bool operator>=(nullptr_t, const unique_ptr<T, D>& y);

  template<class E, class T, class Y, class D>
    basic_ostream<E, T>& operator<<(basic_ostream<E, T>& os, const unique_ptr<Y, D>& p);  
  [...]
  // [util.smartptr.shared.cmp], shared_ptr comparisons
  template<class T, class U>
    bool operator==(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
  template<class T, class U>
    bool operator!=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
  template<class T, class U>
    bool operator<(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
  template<class T, class U>
    bool operator>(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
  template<class T, class U>
    bool operator<=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
  template<class T, class U>
    bool operator>=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;

  template<class T>
    bool operator==(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator==(nullptr_t, const shared_ptr<T>& y) noexcept;
  template<class T>
    bool operator!=(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator!=(nullptr_t, const shared_ptr<T>& y) noexcept;
  template<class T>
    bool operator<(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator<(nullptr_t, const shared_ptr<T>& y) noexcept;
  template<class T>
    bool operator>(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator>(nullptr_t, const shared_ptr<T>& y) noexcept;
  template<class T>
    bool operator<=(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator<=(nullptr_t, const shared_ptr<T>& y) noexcept;
  template<class T>
    bool operator>=(const shared_ptr<T>& x, nullptr_t) noexcept;
  template<class T>
    bool operator>=(nullptr_t, const shared_ptr<T>& y) noexcept;

  // [util.smartptr.shared.spec], shared_ptr specialized algorithms
  template<class T>
    void swap(shared_ptr<T>& a, shared_ptr<T>& b) noexcept;
  [...]    
}

Change 19.10.10 [default.allocator]:

namespace std {
  template<class T> class allocator {
   public:
    using value_type      = T;
    using size_type       = size_t;
    using difference_type = ptrdiff_t;
    using propagate_on_container_move_assignment = true_type;
    using is_always_equal = true_type;

    constexpr allocator() noexcept;
    constexpr allocator(const allocator&) noexcept;
    template<class U> constexpr allocator(const allocator<U>&) noexcept;
    ~allocator();
    allocator& operator=(const allocator&) = default;

    [[nodiscard]] T* allocate(size_t n);
    void deallocate(T* p, size_t n);

    template<class U>
      friend bool operator==(const allocator&, const allocator<U>&) { return true; }
  };
}

Remove 19.10.10.2 [allocator.globals]:

template<class T, class U>
  bool operator==(const allocator<T>&, const allocator<U>&) noexcept;
Returns: true.
template<class T, class U>
  bool operator!=(const allocator<T>&, const allocator<U>&) noexcept;
Returns: false.

Change 19.11.1.2 [unique.ptr.single]:

namespace std {
  template<class T, class D = default_delete<T>> class unique_ptr {
  public:
    using pointer      = see below;
    using element_type = T;
    using deleter_type = D;
    [...]
    // disable copy from lvalue
    unique_ptr(const unique_ptr&) = delete;
    unique_ptr& operator=(const unique_ptr&) = delete;

    // [unique.ptr.special] Specialized algorithms
    template<class T2, class D2>
      friend bool operator==(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return x.get() == y.get(); }
    template<class T2, class D2>
      friend bool operator<(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator<=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator>=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
        requires ThreeWayComparableWith<pointer, typename unique_ptr<T2, D2>::pointer>
      friend auto operator<=>(const unique_ptr& x, const unique_ptr<T2, D2>& y)
      { return compare_three_way()(x.get(), y.get()); }

    friend bool operator==(const unique_ptr& x, nullptr_t) noexcept { return !x; }
    friend auto operator<=>(const unique_ptr& x, nullptr_t)
      requires ThreeWayComparableWith<pointer, nullptr_t>
      { return compare_three_way()(x.get(), nullptr); }
  };
}

Change 19.11.1.3 [unique.ptr.runtime]:

namespace std {
  template class unique_ptr {
  public:
    [...]
    // disable copy from lvalue
    unique_ptr(const unique_ptr&) = delete;
    unique_ptr& operator=(const unique_ptr&) = delete;

    // [unique.ptr.special] Specialized algorithms
    template<class T2, class D2>
      friend bool operator==(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return x.get() == y.get(); }
    template<class T2, class D2>
      friend bool operator<(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator<=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
      friend bool operator>=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below }
    template<class T2, class D2>
        requires ThreeWayComparableWith<pointer, typename unique_ptr<T2, D2>::pointer>
      friend auto operator<=>(const unique_ptr& x, const unique_ptr<T2, D2>& y)
      { return compare_three_way()(x.get(), y.get()); }

    friend bool operator==(const unique_ptr& x, nullptr_t) noexcept { return !x; }
    friend auto operator<=>(const unique_ptr& x, nullptr_t)
      requires ThreeWayComparableWith<pointer, nullptr_t>
      { return compare_three_way()(x.get(), nullptr); }  
  };
}

Change 19.11.1.5 [unique.ptr.special]:

template<class T, class D> void swap(unique_ptr<T, D>& x, unique_ptr<T, D>& y) noexcept;
Remarks: This function shall not participate in overload resolution unless is_swappable_v<D> is true. Effects: Calls x.swap(y).
template<class T1, class D1, class T2, class D2>
  bool operator==(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Returns: x.get() == y.get().
template<class T1, class D1, class T2, class D2>
  bool operator!=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Returns: x.get() != y.get().
template<class T1, class D1, class T2, class D2>
  friend bool operator<(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Requires: Let CT denote common_type_t<typename unique_ptr<T1, D1>::pointer, typename unique_ptr<T2, D2>::pointer> Then the specialization less<CT> shall be a function object type that induces a strict weak ordering on the pointer values.
Returns: less<CT>()(x.get(), y.get()).
Remarks: If unique_ptr<T1, D1>::pointer is not implicitly convertible to CT or unique_ptr<T2, D2>::pointer is not implicitly convertible to CT, the program is ill-formed.
Mandates: pointer and unique_ptr<T2, D2>::pointer are implicitly convertible to CT.
Remarks: This function is to be found via argument-dependent lookup only.
template<class T1, class D1, class T2, class D2>
  friend bool operator>(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Returns: y < x.
Remarks: This function is to be found via argument-dependent lookup only.
template<class T1, class D1, class T2, class D2>
  friend bool operator<=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Returns: !(y < x).
Remarks: This function is to be found via argument-dependent lookup only.
template<class T1, class D1, class T2, class D2>
  friend bool operator>=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);
Returns: !(x < y).
Remarks: This function is to be found via argument-dependent lookup only.

Change 19.11.3, [util.smartptr.shared]:

namespace std {
  template<class T> class shared_ptr {
    [...]
    // [util.smartptr.shared.obs], observers
    element_type* get() const noexcept;
    T& operator*() const noexcept;
    T* operator->() const noexcept;
    element_type& operator[](ptrdiff_t i) const;
    long use_count() const noexcept;
    explicit operator bool() const noexcept;
    template<class U>
      bool owner_before(const shared_ptr<U>& b) const noexcept;
    template<class U>
      bool owner_before(const weak_ptr<U>& b) const noexcept;

    // [util.smartptr.shared.cmp], shared_ptr comparisons
    template<class U>
      friend bool operator==(const shared_ptr& a, const shared_ptr<U>& b) noexcept
      { return a.get() == b.get(); }
    template<class U>
      friend strong_ordering operator<=>(const shared_ptr& a, const shared_ptr<U>& b) noexcept
      { return compare_three_way()(a.get(), b.get()); }

    friend bool operator==(const shared_ptr& a, nullptr_t) noexcept { return !a; }
    friend strong_ordering operator<=>(const shared_ptr& a, nullptr_t) noexcept
      { return compare_three_way()(a.get(), nullptr); }
  };
}

Remove all of 19.11.3.7 [util.smartptr.shared.cmp]:

template<class T, class U>
  bool operator==(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
Returns: a.get() == b.get().
template<class T, class U>
  bool operator<(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;
Returns: less<>()(a.get(), b.get()).
[...]

Change 19.12.1 [mem.res.syn]:

namespace std::pmr {
  // [mem.res.class], class memory_resource
  class memory_resource;

  bool operator==(const memory_resource& a, const memory_resource& b) noexcept;
  bool operator!=(const memory_resource& a, const memory_resource& b) noexcept;

  // [mem.poly.allocator.class], class template polymorphic_allocator
  template<class Tp> class polymorphic_allocator;

  template<class T1, class T2>
    bool operator==(const polymorphic_allocator<T1>& a,
                    const polymorphic_allocator<T2>& b) noexcept;
  template<class T1, class T2>
    bool operator!=(const polymorphic_allocator<T1>& a,
                    const polymorphic_allocator<T2>& b) noexcept;

  // [mem.res.global], global memory resources
  memory_resource* new_delete_resource() noexcept;
  [...]
}

Change 19.12.2 [mem.res.class]:

namespace std::pmr {
  class memory_resource {
    static constexpr size_t max_align = alignof(max_align_t);   // exposition only

  public:
    memory_resource() = default;
    memory_resource(const memory_resource&) = default;
    virtual ~memory_resource();

    memory_resource& operator=(const memory_resource&) = default;

    [[nodiscard]] void* allocate(size_t bytes, size_t alignment = max_align);
    void deallocate(void* p, size_t bytes, size_t alignment = max_align);

    bool is_equal(const memory_resource& other) const noexcept;

    friend bool operator==(const memory_resource&, const memory_resource&) noexcept { see below }

  private:
    virtual void* do_allocate(size_t bytes, size_t alignment) = 0;
    virtual void do_deallocate(void* p, size_t bytes, size_t alignment) = 0;

    virtual bool do_is_equal(const memory_resource& other) const noexcept = 0;
  };
}

Change 19.12.2.3 [mem.res.eq]:

friend bool operator==(const memory_resource& a, const memory_resource& b) noexcept;
Returns: &a == &b || a.is_equal(b).
bool operator!=(const memory_resource& a, const memory_resource& b) noexcept;
Returns: !(a == b).

Change 19.12.3 [mem.poly.allocator.class]:

namespace std::pmr {
  template<class Tp> class polymorphic_allocator {
    memory_resource* memory_rsrc;     // exposition only

  public:
    [...]

    memory_resource* resource() const;

    template<class T2>
      friend bool operator==(const polymorphic_allocator& a, const polymorphic_allocator<T2>& b)
      { see below }
  };
}

Change 19.12.3.3 [mem.poly.allocator.eq]:

template<class T1, class T2>
  friend bool operator==(const polymorphic_allocator<T1>& a,
                  const polymorphic_allocator<T2>& b) noexcept;
Returns: *a.resource() == *b.resource().
Remarks: This function is to be found via argument-dependent lookup only.
template<class T1, class T2>
  bool operator!=(const polymorphic_allocator<T1>& a,
                  const polymorphic_allocator<T2>& b) noexcept;
Returns: !(a == b).

Change 19.13.1 [allocator.adaptor.syn]:

namespace std {
  // class template scoped allocator adaptor
  template<class OuterAlloc, class... InnerAlloc>
    class scoped_allocator_adaptor;

  // [scoped.adaptor.operators], scoped allocator operators
  template<class OuterA1, class OuterA2, class... InnerAllocs>
    bool operator==(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,
                    const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;
  template<class OuterA1, class OuterA2, class... InnerAllocs>
    bool operator!=(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,
                    const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;
}
namespace std {
  template<class OuterAlloc, class... InnerAllocs>
  class scoped_allocator_adaptor : public OuterAlloc {
    [...]
    scoped_allocator_adaptor select_on_container_copy_construction() const;

    // [scoped.adaptor.operators], scoped allocator operators
    template<class Outer2>
      friend bool operator==(const scoped_allocator_adaptor& a,
                             const scoped_allocator_adaptor<Outer2, InnerAllocs...>& b) noexcept
      {  see below }
  };

  template<class OuterAlloc, class... InnerAllocs>
    scoped_allocator_adaptor(OuterAlloc, InnerAllocs...)
      -> scoped_allocator_adaptor<OuterAlloc, InnerAllocs...>;
}

Change 19.13.5 [scoped.adaptor.operators]:

template<class OuterA1, class OuterA2, class... InnerAllocs>
  friend bool operator==(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,
                  const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;
Returns: If sizeof...(InnerAllocs) is zero,
a.outer_allocator() == b.outer_allocator()
otherwise
a.outer_allocator() == b.outer_allocator() && a.inner_allocator() == b.inner_allocator()
Remarks: This function is to be found via argument-dependent lookup only.
template<class OuterA1, class OuterA2, class... InnerAllocs>
  bool operator!=(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,
                  const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;
Returns: !(a == b).

Change 19.14.1 [functional.syn]

namespace std {
  [...]
  template<class R, class... ArgTypes>
    bool operator==(const function<R(ArgTypes...)>&, nullptr_t) noexcept;
  template<class R, class... ArgTypes>
    bool operator==(nullptr_t, const function<R(ArgTypes...)>&) noexcept;
  template<class R, class... ArgTypes>
    bool operator!=(const function<R(ArgTypes...)>&, nullptr_t) noexcept;
  template<class R, class... ArgTypes>
    bool operator!=(nullptr_t, const function<R(ArgTypes...)>&) noexcept;

  // [func.search], searchers
  [...]
}

Change 19.14.8 [range.cmp]/2 to add <=>:

There is an implementation-defined strict total ordering over all pointer values of a given type. This total ordering is consistent with the partial order imposed by the builtin operators <, >, <=, and >=, and <=>.

Change 19.14.16.2 [func.wrap.func]:

namespace std {
  template<class> class function; // not defined

  template<class R, class... ArgTypes>
  class function<R(ArgTypes...)> {
  public:
    using result_type = R;
    [...]

    // [func.wrap.func.targ], function target access
    const type_info& target_type() const noexcept;
    template<class T>       T* target() noexcept;
    template<class T> const T* target() const noexcept;

    friend bool operator==(const function& f, nullptr_t) noexcept { return !f; }
  };    
  [...]
  // [func.wrap.func.nullptr], Null pointer comparisons
  template<class R, class... ArgTypes>
    bool operator==(const function<R(ArgTypes...)>&, nullptr_t) noexcept;

  template<class R, class... ArgTypes>
    bool operator==(nullptr_t, const function<R(ArgTypes...)>&) noexcept;

  template<class R, class... ArgTypes>
    bool operator!=(const function<R(ArgTypes...)>&, nullptr_t) noexcept;

  template<class R, class... ArgTypes>
    bool operator!=(nullptr_t, const function<R(ArgTypes...)>&) noexcept;
  [...]
}

Remove 19.14.16.2.6 [func.wrap.func.nullptr]:

template<class R, class... ArgTypes>
  bool operator==(const function<R(ArgTypes...)>& f, nullptr_t) noexcept;
template<class R, class... ArgTypes>
  bool operator==(nullptr_t, const function<R(ArgTypes...)>& f) noexcept;
Returns: !f.
template<class R, class... ArgTypes>
  bool operator!=(const function<R(ArgTypes...)>& f, nullptr_t) noexcept;
template<class R, class... ArgTypes>
  bool operator!=(nullptr_t, const function<R(ArgTypes...)>& f) noexcept;
Returns: (bool)f.

Add a new row to 19.15.4.3 [meta.unary.prop], the "Type property predicates" table:

TemplateConditionPreconditions
...
template<class T>
struct has_strong_structural_equality;
The type T has strong structural equality ([class.compare.default]).T shall be a complete type, cv void, or an array of unknown bound.

Change 19.17.2 [type.index.overview]. Note that the relational operators on type_index are based on type_info::before (effectively <). type_info could provide a three-way ordering function, but does not. Since an important motivation for the existence of type_index is to be used as a key in an associative container, we do not want to pessimize < - but do want to provide <=>.

namespace std {
  class type_index {
  public:
    type_index(const type_info& rhs) noexcept;
    bool operator==(const type_index& rhs) const noexcept;
    bool operator!=(const type_index& rhs) const noexcept;
    bool operator< (const type_index& rhs) const noexcept;
    bool operator> (const type_index& rhs) const noexcept;
    bool operator<= (const type_index& rhs) const noexcept;
    bool operator>= (const type_index& rhs) const noexcept;
    strong_ordering operator<=>(const type_index& rhs) const noexcept;
    size_t hash_code() const noexcept;
    const char* name() const noexcept;

  private:
    const type_info* target;    // exposition only
    // Note that the use of a pointer here, rather than a reference,
    // means that the default copy/move constructor and assignment
    // operators will be provided and work as expected.
  };
}

Change 19.17.3 [type.index.members]:

type_index(const type_info& rhs) noexcept;
Effects: Constructs a type_index object, the equivalent of target = &rhs.
bool operator==(const type_index& rhs) const noexcept;
Returns: *target == *rhs.target.
bool operator!=(const type_index& rhs) const noexcept;
Returns: *target != *rhs.target.
bool operator<(const type_index& rhs) const noexcept;
Returns: target->before(*rhs.target).
bool operator>(const type_index& rhs) const noexcept;
Returns: rhs.target->before(*target).
bool operator<=(const type_index& rhs) const noexcept;
Returns: !rhs.target->before(*target).
bool operator>=(const type_index& rhs) const noexcept;
Returns: !target->before(*rhs.target).
strong_ordering operator<=>(const type_index& rhs) const noexcept;
Effects: Equivalent to
if (*target == *rhs.target) return strong_ordering::equal;
if (target->before(*rhs.target)) return strong_ordering::less;
return strong_ordering::greater;
size_t hash_code() const noexcept;
Returns: target->hash_code(). [...]

4.6. Clause 20: Strings library

Changing the operators for basic_string and basic_string_view and adding extra type alises to the char_traits specializations provided by the standard.

Change 20.2.3.1 [char.traits.specializations.char]:

namespace std {
  template<> struct char_traits<char> {
    using char_type  = char;
    using int_type   = int;
    using off_type   = streamoff;
    using pos_type   = streampos;
    using state_type = mbstate_t;
    using comparison_category = strong_ordering;
    [...]
  };
}

Change 20.2.3.2 [char.traits.specializations.char8_t]:

namespace std {
  template<> struct char_traits<char8_t> {
    using char_type = char8_t;
    using int_type = unsigned int;
    using off_type = streamoff;
    using pos_type = u8streampos;
    using state_type = mbstate_t;
    using comparison_category = strong_ordering;
    [...]
  };
}

Change 20.2.3.3 [char.traits.specializations.char16_t]:

namespace std {
  template<> struct char_traits<char16_t> {
    using char_type  = char16_t;
    using int_type   = uint_least16_t;
    using off_type   = streamoff;
    using pos_type   = u16streampos;
    using state_type = mbstate_t;
    using comparison_category = strong_ordering;
    [...]
  };
}

Change 20.2.3.4 [char.traits.specializations.char32_t]

namespace std {
  template<> struct char_traits<char32_t> {
    using char_type  = char32_t;
    using int_type   = uint_least32_t;
    using off_type   = streamoff;
    using pos_type   = u32streampos;
    using state_type = mbstate_t;
    using comparison_category = strong_ordering;
    [...]
  };
}

Change 20.2.3.5 [char.traits.specializations.wchar.t]

namespace std {
  template<> struct char_traits<wchar_t> {
    using char_type  = wchar_t;
    using int_type   = wint_t;
    using off_type   = streamoff;
    using pos_type   = wstreampos;
    using state_type = mbstate_t;
    using comparison_category = strong_ordering;
    [...]
  };
}

Change 20.3.1 [string.syn]:

#include <initializer_list>

namespace std {
  [...]
  template<class charT, class traits, class Allocator>
    bool operator==(const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator==(const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);
  template<class charT, class traits, class Allocator>
    bool operator==(const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);
  template<class charT, class traits, class Allocator>
    bool operator!=(const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator!=(const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);
  template<class charT, class traits, class Allocator>
    bool operator!=(const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);

  template<class charT, class traits, class Allocator>
    bool operator< (const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator< (const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);
  template<class charT, class traits, class Allocator>
    bool operator< (const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);
  template<class charT, class traits, class Allocator>
    bool operator> (const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator> (const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);
  template<class charT, class traits, class Allocator>
    bool operator> (const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);

  template<class charT, class traits, class Allocator>
    bool operator<=(const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator<=(const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);
  template<class charT, class traits, class Allocator>
    bool operator<=(const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);
  template<class charT, class traits, class Allocator>
    bool operator>=(const basic_string<charT, traits, Allocator>& lhs,
                    const basic_string<charT, traits, Allocator>& rhs) noexcept;
  template<class charT, class traits, class Allocator>
    bool operator>=(const basic_string<charT, traits, Allocator>& lhs,
                    const charT* rhs);
  template<class charT, class traits, class Allocator>
    bool operator>=(const charT* lhs,
                    const basic_string<charT, traits, Allocator>& rhs);
  [...]
}

Change 20.3.2 [basic.string]/3. Insert wherever the editor deems appropriate:

namespace std {
  template<class charT, class traits = char_traits<charT>,
           class Allocator = allocator<charT>>
  class basic_string {
    [...]
    friend bool operator==(const basic_string& lhs, const basic_string& rhs) { see below }
    friend bool operator==(const basic_string& lhs, const charT* rhs) { see below }
    friend see below operator<=>(const basic_string& lhs, const basic_string& rhs) { see below }
    friend see below operator<=>(const basic_string& lhs, const charT* rhs) { see below }
    [...]
  };
}

Change 20.3.3.2 [string.cmp].

template<class charT, class traits, class Allocator>
  bool operator==(const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator==(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator==(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

friend bool operator==(const basic_string& lhs, const basic_string& rhs);
friend bool operator==(const basic_string& lhs, const charT* rhs);

template<class charT, class traits, class Allocator>
  bool operator!=(const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator!=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator!=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

Effects: Equivalent to return basic_string_view<charT, traits>(lhs) == basic_string_view<charT, traits>(rhs);
Remarks: This function is to be found via argument-dependent lookup only.

template<class charT, class traits, class Allocator>
  bool operator< (const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator< (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator< (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

template<class charT, class traits, class Allocator>
  bool operator> (const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator> (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator> (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

template<class charT, class traits, class Allocator>
  bool operator<=(const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator<=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator<=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

template<class charT, class traits, class Allocator>
  bool operator>=(const basic_string<charT, traits, Allocator>& lhs,
                  const basic_string<charT, traits, Allocator>& rhs) noexcept;
template<class charT, class traits, class Allocator>
  bool operator>=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);
template<class charT, class traits, class Allocator>
  bool operator>=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);

friend see below operator<=>(const basic_string& lhs, const basic_string& rhs);  
friend see below operator<=>(const basic_string& lhs, const charT* rhs);  

Effects: Let op be the operator. Equivalent to:

return basic_string_view<charT, traits>(lhs) op <=> basic_string_view<charT, traits>(rhs);
Remarks: This function is to be found via argument-dependent lookup only.

Change 20.4.1 [string.view.synop]:

namespace std {
  // [string.view.template], class template basic_string_view
  template<class charT, class traits = char_traits<charT>>
  class basic_string_view;

  // [string.view.comparison], non-member comparison functions
  template<class charT, class traits>
    constexpr bool operator==(basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  template<class charT, class traits>
    constexpr bool operator!=(basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  template<class charT, class traits>
    constexpr bool operator< (basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  template<class charT, class traits>
    constexpr bool operator> (basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  template<class charT, class traits>
    constexpr bool operator<=(basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  template<class charT, class traits>
    constexpr bool operator>=(basic_string_view<charT, traits> x,
                              basic_string_view<charT, traits> y) noexcept;
  // see [string.view.comparison], sufficient additional overloads of comparison functions
  [...]
}

Change 20.4.2 [string.view.template], insert wherever the editor deems appropriate

template<class charT, class traits = char_traits<charT>>
class basic_string_view {
  [...]
  friend constexpr bool operator==(basic_string_view, basic_string_view) noexcept { see below }
  friend constexpr see below operator<=>(basic_string_view, basic_string_view) noexcept { see below }
  [...]
};

Remove the entirety of 20.4.3 [string.view.comparison]. The proposed two hidden friend declarations satisfy the requirements without needing extra wording. Replace it with the following:

friend constexpr bool operator==(basic_string_view lhs, basic_string_view rhs) noexcept;
Returns: lhs.compare(rhs) == 0.
Remarks: This function is to be found via argument-dependent lookup only.
friend constexpr see below operator<=>(basic_string_view, basic_string_view) noexcept;
Let R denote the type traits::comparison_category if it exists, otherwise R is weak_ordering.
Returns: static_cast<R>(lhs.compare(rhs) <=> 0).
Remarks: This function is to be found via argument-dependent lookup only.

4.7. Clause 21: Containers library

Change 21.3.2 [array.syn]:

#include <initializer_list>

namespace std {
  // [array], class template array
  template<class T, size_t N> struct array;

  template<class T, size_t N>
    constexpr bool operator==(const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr bool operator!=(const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr bool operator< (const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr bool operator> (const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr bool operator<=(const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr bool operator>=(const array<T, N>& x, const array<T, N>& y);
  template<class T, size_t N>
    constexpr void swap(array<T, N>& x, array<T, N>& y) noexcept(noexcept(x.swap(y)));
  [...]
}

Change 21.3.3 [deque.syn]:

#include <initializer_list>

namespace std {
  // [deque], class template deque
  template<class T, class Allocator = allocator<T>> class deque;

  template<class T, class Allocator>
    bool operator==(const deque<T, Allocator>& x, const deque<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator!=(const deque<T, Allocator>& x, const deque<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator< (const deque<T, Allocator>& x, const deque<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator> (const deque<T, Allocator>& x, const deque<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator<=(const deque<T, Allocator>& x, const deque<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator>=(const deque<T, Allocator>& x, const deque<T, Allocator>& y);

  template<class T, class Allocator>
    void swap(deque<T, Allocator>& x, deque<T, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Change 21.3.4 [forward_list.syn]:

#include <initializer_list>

namespace std {
  // [forwardlist], class template forwardlist
  template<class T, class Allocator = allocator<T>> class forward_list;

  template<class T, class Allocator>
    bool operator==(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator!=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator< (const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator> (const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator<=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator>=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);

  template<class T, class Allocator>
    void swap(forward_list<T, Allocator>& x, forward_list<T, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Change 21.3.5 [list.syn]:

#include <initializer_list>

namespace std {
  // [list], class template list
  template<class T, class Allocator = allocator<T>> class list;

  template<class T, class Allocator>
    bool operator==(const list<T, Allocator>& x, const list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator!=(const list<T, Allocator>& x, const list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator< (const list<T, Allocator>& x, const list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator> (const list<T, Allocator>& x, const list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator<=(const list<T, Allocator>& x, const list<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator>=(const list<T, Allocator>& x, const list<T, Allocator>& y);

  template<class T, class Allocator>
    void swap(list<T, Allocator>& x, list<T, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Change 21.3.6 [vector.syn]:

#include <initializer_list>

namespace std {
  // [vector], class template vector
  template<class T, class Allocator = allocator<T>> class vector;

  template<class T, class Allocator>
    bool operator==(const vector<T, Allocator>& x, const vector<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator!=(const vector<T, Allocator>& x, const vector<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator< (const vector<T, Allocator>& x, const vector<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator> (const vector<T, Allocator>& x, const vector<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator<=(const vector<T, Allocator>& x, const vector<T, Allocator>& y);
  template<class T, class Allocator>
    bool operator>=(const vector<T, Allocator>& x, const vector<T, Allocator>& y);

  template<class T, class Allocator>
    void swap(vector<T, Allocator>& x, vector<T, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Add to 21.2.1 [container.requirements.general], paragraph 4:

In Tables 62, 63, and 64 X denotes a container class containing objects of type T, a and b denote values of type X, i and j denote values of type (possibly-const) X::iterator, u denotes an identifier, r denotes a non-const value of type X, and rv denotes a non-const rvalue of type X.

Add a row to Table 62 — Container requirements:

ExpressionReturn
type
Operational
semantics
Assertion/note
pre/post-condition
Complexity
i <=> j
strong_ordering if X::iterator meets the random access iterator requirements, otherwise strong_equalityconstant

Add to 21.2.1 [container.requirements.general], paragraph 7:

In the expressions

i == j
i != j
i < j
i <= j
i >= j
i > j
i <=> j
i - j
where i and j denote objects of a container's iterator type, either or both may be replaced by an object of the container's const_iterator type referring to the same element with no change in semantics.

Remove 21.2.1 [container.requirements.general] table 64 - the optional container operations are now just <=> instead of the four relational operators, and will be defined inline following the LWG guidance for flat_map.

Table 64 lists operations that are provided for some types of containers but not others. Those containers for which the listed operations are provided shall implement the semantics described in Table 64 unless otherwise stated. If the iterators passed to lexicographical_compare satisfy the constexpr iterator requirements ([iterator.requirements.general]) then the operations described in Table 64 are implemented by constexpr functions.
ExpressionReturn
type
Operational
semantics
Assertion/note
pre/post-condition
Complexity
a < b
convertible to bool
lexicographical_compare(
  a.begin(), a.end(),
  b.begin(), b.end())
Requires: < is defined for values of T. < is a total ordering relationship.linear
a > b
convertible to bool
b < a
linear
a <= b
convertible to bool
!(a > b)
linear
a >= b
convertible to bool
!(a < b)
linear
[Note: The algorithm lexicographical_compare() is defined in [algorithms]. —end note]

Change 21.3.7.1, paragraph 4 [array.overview]:

namespace std {
  template<class T, size_t N>
  struct array {
    [...]

    constexpr T *       data() noexcept;
    constexpr const T * data() const noexcept;

    friend constexpr bool operator==(const array&, const array&) = default;
    friend constexpr synth-3way-result<value_type> operator<=>(const array& x, const array& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }
  };

  template<class T, class... U>
    array(T, U...) -> array<T, 1 + sizeof...(U)>;
}

Change 21.3.8.1, paragraph 2 [deque.overview]:

namespace std {
  template<class T, class Allocator = allocator<T>>
  class deque {
  public:
    [...]
    void     swap(deque&)
      noexcept(allocator_traits<Allocator>::is_always_equal::value);
    void     clear() noexcept;

    friend bool operator==(const deque& x, const deque& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const deque& x, const deque& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }
  };

  [...]
}

Change 21.3.9.1, paragraph 3 [forwardlist.overview]

namespace std {
  template<class T, class Allocator = allocator<T>>
  class forward_list {
  public:
    [...]
    void sort();
    template<class Compare> void sort(Compare comp);

    void reverse() noexcept;

    friend bool operator==(const forward_list& x, const forward_list& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const forward_list& x, const forward_list& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }
  };

  [...]
}

Change 21.3.10.1, paragraph 2 [list.overview]

namespace std {
  template<class T, class Allocator = allocator<T>>
  class list {
  public:
    [...]
    void sort();
    template<class Compare> void sort(Compare comp);

    void reverse() noexcept;

    friend bool operator==(const list& x, const list& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const list& x, const list& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }
  };

  [...]  
}

Change 21.3.11.1, paragraph 2 [vector.overview]

namespace std {
  template<class T, class Allocator = allocator<T>>
  class vector {
  public:
    [...]
    void     swap(vector&)
      noexcept(allocator_traits<Allocator>::propagate_on_container_swap::value ||
               allocator_traits<Allocator>::is_always_equal::value);
    void     clear() noexcept;

    friend bool operator==(const vector& x, const vector& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const vector& x, const vector& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }   
  };
  [...]
}

Change 21.3.12 [vector.bool]:

namespace std {
  template<class Allocator>
  class vector<bool, Allocator> {
  public:
    [...]
    static void swap(reference x, reference y) noexcept;
    void flip() noexcept;       // flips all bits
    void clear() noexcept;

    friend bool operator==(const vector& x, const vector& y)
      { return ranges::equal(x, y); }
    friend strong_ordering operator<=>(const vector& x, const vector& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), compare_three_way()); }
  };
}

Change 21.4.2 [associative.map.syn]:

#include <initializer_list>

namespace std {
  // [map], class template map
  template<class Key, class T, class Compare = less<Key>,
           class Allocator = allocator<pair<const Key, T>>>
    class map;

  template<class Key, class T, class Compare, class Allocator>
    bool operator==(const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator!=(const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator< (const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator> (const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator<=(const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator>=(const map<Key, T, Compare, Allocator>& x,
                    const map<Key, T, Compare, Allocator>& y);

  template<class Key, class T, class Compare, class Allocator>
    void swap(map<Key, T, Compare, Allocator>& x,
              map<Key, T, Compare, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  template <class Key, class T, class Compare, class Allocator, class Predicate>
    void erase_if(map<Key, T, Compare, Allocator>& c, Predicate pred);

  // [multimap], class template multimap
  template<class Key, class T, class Compare = less<Key>,
           class Allocator = allocator<pair<const Key, T>>>
    class multimap;

  template<class Key, class T, class Compare, class Allocator>
    bool operator==(const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator!=(const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator< (const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator> (const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator<=(const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);
  template<class Key, class T, class Compare, class Allocator>
    bool operator>=(const multimap<Key, T, Compare, Allocator>& x,
                    const multimap<Key, T, Compare, Allocator>& y);

  template<class Key, class T, class Compare, class Allocator>
    void swap(multimap<Key, T, Compare, Allocator>& x,
              multimap<Key, T, Compare, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  template <class Key, class T, class Compare, class Allocator, class Predicate>
    void erase_if(multimap<Key, T, Compare, Allocator>& c, Predicate pred);

  namespace pmr {
    template<class Key, class T, class Compare = less<Key>>
      using map = std::map<Key, T, Compare,
                           polymorphic_allocator<pair<const Key, T>>>;

    template<class Key, class T, class Compare = less<Key>>
      using multimap = std::multimap<Key, T, Compare,
                                     polymorphic_allocator<pair<const Key, T>>>;
  }
}

Change 21.4.3 [associative.set.syn]:

#include <initializer_list>

namespace std {
  // [set], class template set
  template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>>
    class set;

  template<class Key, class Compare, class Allocator>
    bool operator==(const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator!=(const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator< (const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator> (const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator<=(const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator>=(const set<Key, Compare, Allocator>& x,
                    const set<Key, Compare, Allocator>& y);

  template<class Key, class Compare, class Allocator>
    void swap(set<Key, Compare, Allocator>& x,
              set<Key, Compare, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  template <class Key, class Compare, class Allocator, class Predicate>
    void erase_if(set<Key, Compare, Allocator>& c, Predicate pred);

  // [multiset], class template multiset
  template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>>
    class multiset;

  template<class Key, class Compare, class Allocator>
    bool operator==(const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator!=(const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator< (const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator> (const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator<=(const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);
  template<class Key, class Compare, class Allocator>
    bool operator>=(const multiset<Key, Compare, Allocator>& x,
                    const multiset<Key, Compare, Allocator>& y);

  template<class Key, class Compare, class Allocator>
    void swap(multiset<Key, Compare, Allocator>& x,
              multiset<Key, Compare, Allocator>& y)
      noexcept(noexcept(x.swap(y)));

  template <class Key, class Compare, class Allocator, class Predicate>
    void erase_if(multiset<Key, Compare, Allocator>& c, Predicate pred);

  namespace pmr {
    template<class Key, class Compare = less<Key>>
      using set = std::set<Key, Compare, polymorphic_allocator<Key>>;

    template<class Key, class Compare = less<Key>>
      using multiset = std::multiset<Key, Compare, polymorphic_allocator<Key>>;
  }
}

Change 21.4.4.1 [map.overview]:

namespace std {
  template<class Key, class T, class Compare = less<Key>,
           class Allocator = allocator<pair<const Key, T>>>
  class map {
  public:
    [...]
    pair<iterator, iterator>               equal_range(const key_type& x);
    pair<const_iterator, const_iterator>   equal_range(const key_type& x) const;
    template<class K>
      pair<iterator, iterator>             equal_range(const K& x);
    template<class K>
      pair<const_iterator, const_iterator> equal_range(const K& x) const;

    friend bool operator==(const map& x, const map& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const map& x, const map& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }
  };

  [...]
}

Change 21.4.5.1 [multimap.overview]:

namespace std {
  template<class Key, class T, class Compare = less<Key>,
           class Allocator = allocator<pair<const Key, T>>>
  class multimap {
  public:
    [...]
    pair<iterator, iterator>               equal_range(const key_type& x);
    pair<const_iterator, const_iterator>   equal_range(const key_type& x) const;
    template<class K>
      pair<iterator, iterator>             equal_range(const K& x);
    template<class K>
      pair<const_iterator, const_iterator> equal_range(const K& x) const;

    friend bool operator==(const multimap& x, const multimap& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const multimap& x, const multimap& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }

  };

  [...]
}

Change 21.4.6.1 [set.overview]:

namespace std {
  template<class Key, class Compare = less<Key>,
           class Allocator = allocator<Key>>
  class set {
  public:
    [...]
    pair<iterator, iterator>               equal_range(const key_type& x);
    pair<const_iterator, const_iterator>   equal_range(const key_type& x) const;
    template<class K>
      pair<iterator, iterator>             equal_range(const K& x);
    template<class K>
      pair<const_iterator, const_iterator> equal_range(const K& x) const;

    friend bool operator==(const set& x, const set& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const set& x, const set& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }          
  };

  [...]
}

Change 21.4.7.1 [multiset.overview]:

namespace std {
  template<class Key, class Compare = less<Key>,
           class Allocator = allocator<Key>>
  class multiset {
  public:
    [...]

    pair<iterator, iterator>               equal_range(const key_type& x);
    pair<const_iterator, const_iterator>   equal_range(const key_type& x) const;
    template<class K>
      pair<iterator, iterator>             equal_range(const K& x);
    template<class K>
      pair<const_iterator, const_iterator> equal_range(const K& x) const;

    friend bool operator==(const multiset& x, const multiset& y)
      { return ranges::equal(x, y); }
    friend synth-3way-result<value_type> operator<=>(const multiset& x, const multiset& y)
      { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); }      
  };

  [...]
}

Change 21.5.2 [unord.map.syn]:

#include <initializer_list>

namespace std {
  // [unord.map], class template unordered_map
  template<class Key,
           class T,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Alloc = allocator<pair<const Key, T>>>
    class unordered_map;

  // [unord.multimap], class template unordered_multimap
  template<class Key,
           class T,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Alloc = allocator<pair<const Key, T>>>
    class unordered_multimap;

  template<class Key, class T, class Hash, class Pred, class Alloc>
    bool operator==(const unordered_map<Key, T, Hash, Pred, Alloc>& a,
                    const unordered_map<Key, T, Hash, Pred, Alloc>& b);
  template<class Key, class T, class Hash, class Pred, class Alloc>
    bool operator!=(const unordered_map<Key, T, Hash, Pred, Alloc>& a,
                    const unordered_map<Key, T, Hash, Pred, Alloc>& b);

  template<class Key, class T, class Hash, class Pred, class Alloc>
    bool operator==(const unordered_multimap<Key, T, Hash, Pred, Alloc>& a,
                    const unordered_multimap<Key, T, Hash, Pred, Alloc>& b);
  template<class Key, class T, class Hash, class Pred, class Alloc>
    bool operator!=(const unordered_multimap<Key, T, Hash, Pred, Alloc>& a,
                    const unordered_multimap<Key, T, Hash, Pred, Alloc>& b);

  template<class Key, class T, class Hash, class Pred, class Alloc>
    void swap(unordered_map<Key, T, Hash, Pred, Alloc>& x,
              unordered_map<Key, T, Hash, Pred, Alloc>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Change 21.5.3 [unord.set.syn]:

#include <initializer_list>

namespace std {
  // [unord.set], class template unordered_set
  template<class Key,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Alloc = allocator<Key>>
    class unordered_set;

  // [unord.multiset], class template unordered_multiset
  template<class Key,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Alloc = allocator<Key>>
    class unordered_multiset;

  template<class Key, class Hash, class Pred, class Alloc>
    bool operator==(const unordered_set<Key, Hash, Pred, Alloc>& a,
                    const unordered_set<Key, Hash, Pred, Alloc>& b);
  template<class Key, class Hash, class Pred, class Alloc>
    bool operator!=(const unordered_set<Key, Hash, Pred, Alloc>& a,
                    const unordered_set<Key, Hash, Pred, Alloc>& b);

  template<class Key, class Hash, class Pred, class Alloc>
    bool operator==(const unordered_multiset<Key, Hash, Pred, Alloc>& a,
                    const unordered_multiset<Key, Hash, Pred, Alloc>& b);
  template<class Key, class Hash, class Pred, class Alloc>
    bool operator!=(const unordered_multiset<Key, Hash, Pred, Alloc>& a,
                    const unordered_multiset<Key, Hash, Pred, Alloc>& b);

  template<class Key, class Hash, class Pred, class Alloc>
    void swap(unordered_set<Key, Hash, Pred, Alloc>& x,
              unordered_set<Key, Hash, Pred, Alloc>& y)
      noexcept(noexcept(x.swap(y)));

  [...]
}

Change 21.5.4.1 [unord.map.overview]:

namespace std {
  template<class Key,
           class T,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Allocator = allocator<pair<const Key, T>>>
  class unordered_map {
  public:
    [...]
    // hash policy
    float load_factor() const noexcept;
    float max_load_factor() const noexcept;
    void max_load_factor(float z);
    void rehash(size_type n);
    void reserve(size_type n);

    friend bool operator==(const unordered_map& x, const unordered_map& y)
      { return ranges::equal(x, y); }
  };

  [...]
}

Change 21.5.5.1 [unord.multimap.overview]:

namespace std {
  template<class Key,
           class T,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Allocator = allocator<pair<const Key, T>>>
  class unordered_multimap {
  public:
    [...]
    // hash policy
    float load_factor() const noexcept;
    float max_load_factor() const noexcept;
    void max_load_factor(float z);
    void rehash(size_type n);
    void reserve(size_type n);

    friend bool operator==(const unordered_multimap& x, const unordered_multimap& y)
      { return ranges::equal(x, y); }    
  };

  [...]
}

Change 21.5.6.1 [unord.set.overview]:

namespace std {
  template<class Key,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Allocator = allocator<Key>>
  class unordered_set {
  public:
    [...]
    // hash policy
    float load_factor() const noexcept;
    float max_load_factor() const noexcept;
    void max_load_factor(float z);
    void rehash(size_type n);
    void reserve(size_type n);

    friend bool operator==(const unordered_set& x, const unordered_set& y)
      { return ranges::equal(x, y); }        
  };

  [...]
}

Change 21.5.7.1 [unord.multiset.overview]:

namespace std {
  template<class Key,
           class Hash = hash<Key>,
           class Pred = equal_to<Key>,
           class Allocator = allocator<Key>>
  class unordered_multiset {
  public:
    [...]
    // hash policy
    float load_factor() const noexcept;
    float max_load_factor() const noexcept;
    void max_load_factor(float z);
    void rehash(size_type n);
    void reserve(size_type n);

    friend bool operator==(const unordered_multiset& x, const unordered_multiset& y)
      { return ranges::equal(x, y); }        
  };

  [...]
}

Change 21.6.2 [queue.syn]:

#include <initializer_list>

namespace std {
  template<class T, class Container = deque<T>> class queue;
  template<class T, class Container = vector<T>,
           class Compare = less<typename Container::value_type>>
    class priority_queue;

  template<class T, class Container>
    bool operator==(const queue<T, Container>& x, const queue<T, Container>& y);
  template<class T, class Container>
    bool operator!=(const queue<T, Container>& x, const queue<T, Container>& y);
  template<class T, class Container>
    bool operator< (const queue<T, Container>& x, const queue<T, Container>& y);
  template<class T, class Container>
    bool operator> (const queue<T, Container>& x, const queue<T, Container>& y);
  template<class T, class Container>
    bool operator<=(const queue<T, Container>& x, const queue<T, Container>& y);
  template<class T, class Container>
    bool operator>=(const queue<T, Container>& x, const queue<T, Container>& y);

  template<class T, class Container>
    void swap(queue<T, Container>& x, queue<T, Container>& y) noexcept(noexcept(x.swap(y)));
  template<class T, class Container, class Compare>
    void swap(priority_queue<T, Container, Compare>& x,
              priority_queue<T, Container, Compare>& y) noexcept(noexcept(x.swap(y)));
}

Change 21.6.3 [stack.syn]:

#include <initializer_list>

namespace std {
  template<class T, class Container = deque<T>> class stack;

  template<class T, class Container>
    bool operator==(const stack<T, Container>& x, const stack<T, Container>& y);
  template<class T, class Container>
    bool operator!=(const stack<T, Container>& x, const stack<T, Container>& y);
  template<class T, class Container>
    bool operator< (const stack<T, Container>& x, const stack<T, Container>& y);
  template<class T, class Container>
    bool operator> (const stack<T, Container>& x, const stack<T, Container>& y);
  template<class T, class Container>
    bool operator<=(const stack<T, Container>& x, const stack<T, Container>& y);
  template<class T, class Container>
    bool operator>=(const stack<T, Container>& x, const stack<T, Container>& y);

  template<class T, class Container>
    void swap(stack<T, Container>& x, stack<T, Container>& y) noexcept(noexcept(x.swap(y)));
}

Change 21.6.4.1 [queue.defn]:

namespace std {
  template<class T, class Container = deque<T>>
  class queue {
  public:
    using value_type      = typename Container::value_type;
    using reference       = typename Container::reference;
    using const_reference = typename Container::const_reference;
    using size_type       = typename Container::size_type;
    using container_type  =          Container;

  protected:
    Container c;

  public:
    queue() : queue(Container()) {}
    explicit queue(const Container&);
    explicit queue(Container&&);
    template<class Alloc> explicit queue(const Alloc&);
    template<class Alloc> queue(const Container&, const Alloc&);
    template<class Alloc> queue(Container&&, const Alloc&);
    template<class Alloc> queue(const queue&, const Alloc&);
    template<class Alloc> queue(queue&&, const Alloc&);

    [[nodiscard]] bool empty() const    { return c.empty(); }
    size_type         size()  const     { return c.size(); }
    reference         front()           { return c.front(); }
    const_reference   front() const     { return c.front(); }
    reference         back()            { return c.back(); }
    const_reference   back() const      { return c.back(); }
    void push(const value_type& x)      { c.push_back(x); }
    void push(value_type&& x)           { c.push_back(std::move(x)); }
    template<class... Args>
      decltype(auto) emplace(Args&&... args)
        { return c.emplace_back(std::forward<Args>(args)...); }
    void pop()                          { c.pop_front(); }
    void swap(queue& q) noexcept(is_nothrow_swappable_v<Container>)
      { using std::swap; swap(c, q.c); }

    friend bool operator==(const queue& x, const queue& y)
      { return x.c == y.c; }
    friend bool operator!=(const queue& x, const queue& y)
      { return x.c != y.c; }
    friend bool operator< (const queue& x, const queue& y)
      { return x.c < y.c; }
    friend bool operator> (const queue& x, const queue& y)
      { return x.c > y.c; }
    friend bool operator<=(const queue& x, const queue& y)
      { return x.c <= y.c; }
    friend bool operator>=(const queue& x, const queue& y)
      { return x.c >= y.c; }
    friend auto operator<=>(const queue& x, const queue& y)
      requires ThreeWayComparable<Container>
        { return x.c <=> y.c; }
  };
  [...]
}

Remove 21.6.4.4 [queue.ops] (as we've now defined them all inline in the header):

template<class T, class Container>
  bool operator==(const queue<T, Container>& x, const queue<T, Container>& y);
Returns: x.c == y.c.
[...]

Change 21.6.6.1 [stack.defn]:

namespace std {
  template<class T, class Container = deque<T>>
  class stack {
  public:
    using value_type      = typename Container::value_type;
    using reference       = typename Container::reference;
    using const_reference = typename Container::const_reference;
    using size_type       = typename Container::size_type;
    using container_type  = Container;

  protected:
    Container c;

  public:
    stack() : stack(Container()) {}
    explicit stack(const Container&);
    explicit stack(Container&&);
    template<class Alloc> explicit stack(const Alloc&);
    template<class Alloc> stack(const Container&, const Alloc&);
    template<class Alloc> stack(Container&&, const Alloc&);
    template<class Alloc> stack(const stack&, const Alloc&);
    template<class Alloc> stack(stack&&, const Alloc&);

    [[nodiscard]] bool empty() const    { return c.empty(); }
    size_type size()  const             { return c.size(); }
    reference         top()             { return c.back(); }
    const_reference   top() const       { return c.back(); }
    void push(const value_type& x)      { c.push_back(x); }
    void push(value_type&& x)           { c.push_back(std::move(x)); }
    template<class... Args>
      decltype(auto) emplace(Args&&... args)
        { return c.emplace_back(std::forward<Args>(args)...); }
    void pop()                          { c.pop_back(); }
    void swap(stack& s) noexcept(is_nothrow_swappable_v<Container>)
      { using std::swap; swap(c, s.c); }

    friend bool operator==(const stack& x, const stack& y)
      { return x.c == y.c; }
    friend bool operator!=(const stack& x, const stack& y)
      { return x.c != y.c; }
    friend bool operator< (const stack& x, const stack& y)
      { return x.c < y.c; }
    friend bool operator> (const stack& x, const stack& y)
      { return x.c > y.c; }
    friend bool operator<=(const stack& x, const stack& y)
      { return x.c <= y.c; }
    friend bool operator>=(const stack& x, const stack& y)
      { return x.c >= y.c; }
    friend auto operator<=>(const stack& x, const stack& y)
      requires ThreeWayComparable<Container>
        { return x.c <=> y.c; }      
  };

  [...]
}

Remove 21.6.6.4 [stack.ops] (as we've now defined them all inline in the header):

template<class T, class Container>
  bool operator==(const stack<T, Container>& x, const stack<T, Container>& y);
Returns: x.c == y.c.
[...]

4.8. Clause 22: Iterators library

Change 22.2 [iterator.synopsis]:

#include <concepts>

namespace std {
  [...]
  // [predef.iterators], predefined iterators and sentinels
  // [reverse.iterators], reverse iterators
  template<class Iterator> class reverse_iterator;

  template<class Iterator1, class Iterator2>
    constexpr bool operator==(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator!=(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator<(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator>(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator<=(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator>=(
      const reverse_iterator<Iterator1>& x,
      const reverse_iterator<Iterator2>& y);
  [...]

  // [move.iterators], move iterators and sentinels
  template<class Iterator> class move_iterator;

  template<class Iterator1, class Iterator2>
    constexpr bool operator==(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator!=(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator<(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator>(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator<=(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
  template<class Iterator1, class Iterator2>
    constexpr bool operator>=(
      const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);

  [...]

  // [stream.iterators], stream iterators
  template<class T, class charT = char, class traits = char_traits<charT>,
           class Distance = ptrdiff_t>
  class istream_iterator;
  template<class T, class charT, class traits, class Distance>
    bool operator==(const istream_iterator<T,charT,traits,Distance>& x,
            const istream_iterator<T,charT,traits,Distance>& y);
  template<class T, class charT, class traits, class Distance>
    bool operator!=(const istream_iterator<T,charT,traits,Distance>& x,
            const istream_iterator<T,charT,traits,Distance>& y);

  template<class T, class charT = char, class traits = char_traits<charT>>
      class ostream_iterator;

  template<class charT, class traits = char_traits<charT>>
    class istreambuf_iterator;
  template<class charT, class traits>
    bool operator==(const istreambuf_iterator<charT,traits>& a,
            const istreambuf_iterator<charT,traits>& b);
  template<class charT, class traits>
    bool operator!=(const istreambuf_iterator<charT,traits>& a,
            const istreambuf_iterator<charT,traits>& b);

  template<class charT, class traits = char_traits<charT>>
    class ostreambuf_iterator;  
  [...]
}

Change 22.5.1.1 [reverse.iterator]:

namespace std {
  template<class Iterator>
  class reverse_iterator {
  public:
    [...]
    template<IndirectlySwappable<Iterator> Iterator2>
      friend constexpr void
        iter_swap(const reverse_iterator& x,
                  const reverse_iterator<Iterator2>& y) noexcept(see below);

    // [reverse.iter.cmp] Comparisons
    template<class Iterator2>
      friend constexpr bool operator==(const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator!=(const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator< (const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator> (const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator<=(const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator>=(const reverse_iterator& x,
                                       const reverse_iterator<Iterator2>& y)
      { see below }
    template<ThreeWayComparableWith<Iterator> Iterator2>
      friend constexpr compare_three_way_result_t<Iterator, Iterator2>
        operator<=>(const reverse_iterator& x,
                    const reverse_iterator<Iterator2>& y)
        { see below }

  protected:
    Iterator current;
  };
}

Change 22.5.1.7 [reverse.iter.cmp]:

template<class Iterator1, class Iterator2>
  friend constexpr bool operator==(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current == y.current is well-formed and convertible to bool.
Returns: x.current == y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  friend constexpr bool operator!=(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current != y.current is well-formed and convertible to bool.
Returns: x.current != y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  friend constexpr bool operator<(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current > y.current is well-formed and convertible to bool.
Returns: x.current > y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  friend constexpr bool operator>(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current < y.current is well-formed and convertible to bool.
Returns: x.current < y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  friend constexpr bool operator<=(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current >= y.current is well-formed and convertible to bool.
Returns: x.current >= y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  friend constexpr bool operator>=(
    const reverse_iterator<Iterator1>& x,
    const reverse_iterator<Iterator2>& y);
Constraints: x.current <= y.current is well-formed and convertible to bool.
Returns: x.current <= y.current.
Remarks: This function is to be found via argument-dependent lookup only.
template<ThreeWayComparableWith<Iterator> Iterator2>
  friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>(
    const reverse_iterator& x,
    const reverse_iterator<Iterator2>& y);
Returns: y.current <=> x.current.
Remarks: This function is more constrained than ([temp.constr.order]) each of the other relational operator function templates. This function is to be found via argument-dependent lookup only.

Change 22.5.3.1 [move.iterator]:

namespace std {
  template<class Iterator>
  class move_iterator {
    [...]
    // [move.iter.op.comp] Comparisons
    template<class Iterator2>
      friend constexpr bool operator==(const move_iterator& x, const move_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator<(const move_iterator& x, const move_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator>(const move_iterator& x, const move_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator<=(const move_iterator& x, const move_iterator<Iterator2>& y)
      { see below }
    template<class Iterator2>
      friend constexpr bool operator>=(const move_iterator& x, const move_iterator<Iterator2>& y)
      { see below }
    template<ThreeWayComparableWith<Iterator> Iterator2>
      friend constexpr compare_three_way_result_t<Iterator, Iterator2>
        operator<=>(const move_iterator& x, const move_iterator<Iterator2>& y)
        { see below }

    template<Sentinel<Iterator> S>
      friend constexpr bool
        operator==(const move_iterator& x, const move_sentinel<S>& y);
    template<Sentinel<Iterator> S>
      friend constexpr bool
        operator==(const move_sentinel<S>& x, const move_iterator& y);
    template<Sentinel<Iterator> S>
      friend constexpr bool
        operator!=(const move_iterator& x, const move_sentinel<S>& y);
    template<Sentinel<Iterator> S>
      friend constexpr bool
        operator!=(const move_sentinel<S>& x, const move_iterator& y);
    template<SizedSentinel<Iterator> S>
      friend constexpr iter_difference_t<Iterator>
        operator-(const move_sentinel<S>& x, const move_iterator& y);
    [...]
  private:
    Iterator current;   // exposition only
  };
}

Change 22.5.3.7 [move.iter.op.comp]:

template<class Iterator1, class Iterator2>
  friend constexpr bool operator==(const move_iterator<Iterator1>& x,
                            const move_iterator<Iterator2>& y);
template<Sentinel<Iterator> S>
  friend constexpr bool operator==(const move_iterator& x,
                                   const move_sentinel<S>& y);
template<Sentinel<Iterator> S>
  friend constexpr bool operator==(const move_sentinel<S>& x,
                                   const move_iterator& y);
Constraints: x.base() == y.base() is well-formed and convertible to bool.
Returns: x.base() == y.base().
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
  constexpr bool operator!=(const move_iterator<Iterator1>& x,
                            const move_iterator<Iterator2>& y);
template<Sentinel<Iterator> S>
  friend constexpr bool operator!=(const move_iterator& x,
                                   const move_sentinel<S>& y);
template<Sentinel<Iterator> S>
  friend constexpr bool operator!=(const move_sentinel<S>& x,
                                   const move_iterator& y);
Constraints: x.base() == y.base() is well-formed and convertible to bool.
Returns: !(x == y).
template<class Iterator1, class Iterator2>
friend constexpr bool operator<(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
Constraints: x.base() < y.base() is well-formed and convertible to bool.
Returns: x.base() < y.base().
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
friend constexpr bool operator>(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
Constraints: y.base() < x.base() is well-formed and convertible to bool.
Returns: y < x.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
friend constexpr bool operator<=(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
Constraints: y.base() < x.base() is well-formed and convertible to bool.
Returns: !(y < x).
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2>
friend constexpr bool operator>=(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);
Constraints: x.base() < y.base() is well-formed and convertible to bool.
Returns: !(x < y).
Remarks: This function is to be found via argument-dependent lookup only.
template<ThreeWayComparableWith<Iterator> Iterator2>
  friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>(
    const move_iterator& x,
    const move_iterator<Iterator2>& y);
Returns: x.base() <=> y.base().
Remarks: This function is more constrained than ([temp.constr.order]) each of the other relational operator function templates. This function is to be found via argument-dependent lookup only.

Remove the operator!= from 22.5.4.1 [common.iterator]:

namespace std {
  template<Iterator I, Sentinel<I> S>
    requires (!Same<I, S>)
  class common_iterator {
  public:
    [...]
    template<class I2, Sentinel<I> S2>
      requires Sentinel<S, I2>
    friend bool operator==(
      const common_iterator& x, const common_iterator<I2, S2>& y);
    template<class I2, Sentinel<I> S2>
      requires Sentinel<S, I2> && EqualityComparableWith<I, I2>
    friend bool operator==(
      const common_iterator& x, const common_iterator<I2, S2>& y);
    template<class I2, Sentinel<I> S2>
      requires Sentinel<S, I2>
    friend bool operator!=(
      const common_iterator& x, const common_iterator<I2, S2>& y);
    [...]
  private:
    variant<I, S> v_;   // exposition only
  };
  [...]
}

Remove the operator!= from 22.5.4.6 [common.iter.cmp]:

template<class I2, Sentinel<I> S2>
  requires Sentinel<S, I2>
friend bool operator!=(
  const common_iterator& x, const common_iterator<I2, S2>& y);
Effects: Equivalent to: return !(x == y);

Change 22.5.6.1 [counted.iterator]:

namespace std {
  template<Iterator I>
  class counted_iterator {
  public:
    using iterator_type = I;
    [...]
    template<Common<I> I2>
      friend constexpr bool operator==(
        const counted_iterator& x, const counted_iterator<I2>& y);
    friend constexpr bool operator==(
      const counted_iterator& x, default_sentinel_t);
    friend constexpr bool operator==(
      default_sentinel_t, const counted_iterator& x);

    template<Common<I> I2>
      friend constexpr bool operator!=(
        const counted_iterator& x, const counted_iterator<I2>& y);
    friend constexpr bool operator!=(
      const counted_iterator& x, default_sentinel_t y);
    friend constexpr bool operator!=(
      default_sentinel_t x, const counted_iterator& y);

    template<Common<I> I2>
      friend constexpr bool operator<(
        const counted_iterator& x, const counted_iterator<I2>& y);
    template<Common<I> I2>
      friend constexpr bool operator>(
        const counted_iterator& x, const counted_iterator<I2>& y);
    template<Common<I> I2>
      friend constexpr bool operator<=(
        const counted_iterator& x, const counted_iterator<I2>& y);
    template<Common<I> I2>
      friend constexpr bool operator>=(
        const counted_iterator& x, const counted_iterator<I2>& y);
    template<Common<I> I2> requires ThreeWayComparableWith<I, I2>
      friend constexpr compare_three_way_result_t<I, I2> operator<=>(
        const counted_iterator& x, const counted_iterator<I2>& y);

    friend constexpr iter_rvalue_reference_t<I> iter_move(const counted_iterator& i)
      noexcept(noexcept(ranges::iter_move(i.current)))
        requires InputIterator<I>;
    template<IndirectlySwappable<I> I2>
      friend constexpr void iter_swap(const counted_iterator& x, const counted_iterator<I2>& y)
        noexcept(noexcept(ranges::iter_swap(x.current, y.current)));

  private:
    I current = I();                    // exposition only
    iter_difference_t<I> length = 0;    // exposition only
  };
  [...]
}

Change 22.5.6.6 [counted.iter.cmp]:

template<Common<I> I2>
  friend constexpr bool operator==(
    const counted_iterator& x, const counted_iterator<I2>& y);
Expects: x and y refer to elements of the same sequence ([counted.iterator]).
Effects: Equivalent to: return x.length == y.length;
friend constexpr bool operator==(
  const counted_iterator& x, default_sentinel_t);
friend constexpr bool operator==(
  default_sentinel_t, const counted_iterator& x);
Effects: Equivalent to: return x.length == 0;
template<Common<I> I2>
  friend constexpr bool operator!=(
    const counted_iterator& x, const counted_iterator<I2>& y);
friend constexpr bool operator!=(
  const counted_iterator& x, default_sentinel_t y);
friend constexpr bool operator!=(
  default_sentinel_t x, const counted_iterator& y);
Effects: Equivalent to: return !(x == y);
template<Common<I> I2>
  friend constexpr bool operator<(
    const counted_iterator& x, const counted_iterator<I2>& y);
Expects: x and y refer to elements of the same sequence ([counted.iterator]).
Effects: Equivalent to: return y.length < x.length;
[Note: The argument order in the Effects: element is reversed because length counts down, not up. —end note]
template<Common<I> I2>
  friend constexpr bool operator>(
    const counted_iterator& x, const counted_iterator<I2>& y);
Effects: Equivalent to: return y < x;
template<Common<I> I2>
  friend constexpr bool operator<=(
    const counted_iterator& x, const counted_iterator<I2>& y);
Effects: Equivalent to: return !(y < x);
template<Common<I> I2>
  friend constexpr bool operator>=(
    const counted_iterator& x, const counted_iterator<I2>& y);
Effects: Equivalent to: return !(x < y);
template<Common<I> I2> requires ThreeWayComparableWith<I, I2>
  friend constexpr compare_three_way_result_t<I, I2> operator<=>(
    const counted_iterator& x, const counted_iterator<I2>& y);
Effects: Equivalent to: return y <=> x;

Change 22.5.7.1 [unreachable.sentinel]:

namespace std {
  struct unreachable_sentinel_t {
    template<WeaklyIncrementable I>
      friend constexpr bool operator==(unreachable_sentinel_t, const I&) noexcept;
      { return true; }
    template<WeaklyIncrementable I>
      friend constexpr bool operator==(const I&, unreachable_sentinel_t) noexcept;
    template<WeaklyIncrementable I>
      friend constexpr bool operator!=(unreachable_sentinel_t, const I&) noexcept;
    template<WeaklyIncrementable I>
      friend constexpr bool operator!=(const I&, unreachable_sentinel_t) noexcept;
  };
}

Remove 22.5.7.2 [unreachable.sentinel.cmp] (as it's now entirely defined in the synopsis):

template<WeaklyIncrementable I>
  friend constexpr bool operator==(unreachable_sentinel_t, const I&) noexcept;
template<WeaklyIncrementable I>
  friend constexpr bool operator==(const I&, unreachable_sentinel_t) noexcept;
Returns: false.
template<WeaklyIncrementable I>
  friend constexpr bool operator!=(unreachable_sentinel_t, const I&) noexcept;
template<WeaklyIncrementable I>
  friend constexpr bool operator!=(const I&, unreachable_sentinel_t) noexcept;
Returns: true.

Change 22.6.1 [istream.iterator]:

namespace std {
  template<class T, class charT = char, class traits = char_traits<charT>,
           class Distance = ptrdiff_t>
  class istream_iterator {
  public:
    using iterator_category = input_iterator_tag;
    using value_type        = T;
    using difference_type   = Distance;
    using pointer           = const T*;
    using reference         = const T&;
    using char_type         = charT;
    using traits_type       = traits;
    using istream_type      = basic_istream<charT,traits>;

    constexpr istream_iterator();
    constexpr istream_iterator(default_sentinel_t);
    istream_iterator(istream_type& s);
    istream_iterator(const istream_iterator& x) = default;
    ~istream_iterator() = default;
    istream_iterator& operator=(const istream_iterator&) = default;

    const T& operator*() const;
    const T* operator->() const;
    istream_iterator& operator++();
    istream_iterator  operator++(int);

    friend bool operator==(const istream_iterator& i, default_sentinel_t);
    { return !i.in_stream; }
    friend bool operator==(const istream_iterator& x, const istream_iterator& y)
    { return x.in_stream == y.in_stream; }
    friend bool operator==(default_sentinel_t, const istream_iterator& i);
    friend bool operator!=(const istream_iterator& x, default_sentinel_t y);
    friend bool operator!=(default_sentinel_t x, const istream_iterator& y);

  private:
    basic_istream<charT,traits>* in_stream; // exposition only
    T value;                                // exposition only
  };
}

Remove the specifications for operator== and operator!= from 22.6.1.2 [istream.iterator.ops]:

template<class T, class charT, class traits, class Distance>
  bool operator==(const istream_iterator<T,charT,traits,Distance>& x,
                  const istream_iterator<T,charT,traits,Distance>& y);
Returns: x.in_stream == y.in_stream.
friend bool operator==(default_sentinel_t, const istream_iterator& i);
friend bool operator==(const istream_iterator& i, default_sentinel_t);
Returns: !i.in_stream.
template<class T, class charT, class traits, class Distance>
  bool operator!=(const istream_iterator<T,charT,traits,Distance>& x,
                  const istream_iterator<T,charT,traits,Distance>& y);
friend bool operator!=(default_sentinel_t x, const istream_iterator& y);
friend bool operator!=(const istream_iterator& x, default_sentinel_t y);
Returns: !(x == y)

Change 22.6.3 [istreambuf.iterator]:

namespace std {
  template<class charT, class traits = char_traits<charT>>
  class istreambuf_iterator {
  public:
    [...]
    bool equal(const istreambuf_iterator& b) const;

    friend bool operator==(default_sentinel_t s, const istreambuf_iterator& i);
    friend bool operator==(const istreambuf_iterator& i, default_sentinel_t s);
    { return i.equal(s); }
    friend bool operator!=(default_sentinel_t a, const istreambuf_iterator& b);
    friend bool operator!=(const istreambuf_iterator& a, default_sentinel_t b);
    friend bool operator==(const istreambuf_iterator& a, const istreambuf_iterator& b)
    { return a.equal(b); }

  private:
    streambuf_type* sbuf_;                // exposition only
  };
}

Remove the specifications for operator== and operator!= from 22.6.3.3 [istreambuf.iterator.ops]:

template<class charT, class traits>
  bool operator==(const istreambuf_iterator<charT,traits>& a,
                  const istreambuf_iterator<charT,traits>& b);
Returns: a.equal(b).
friend bool operator==(default_sentinel_t s, const istreambuf_iterator& i);
friend bool operator==(const istreambuf_iterator& i, default_sentinel_t s);
Returns: i.equal(s).
template<class charT, class traits>
  bool operator!=(const istreambuf_iterator<charT,traits>& a,
                  const istreambuf_iterator<charT,traits>& b);
friend bool operator!=(default_sentinel_t a, const istreambuf_iterator& b);
friend bool operator!=(const istreambuf_iterator& a, default_sentinel_t b);
Returns: !a.equal(b).

4.9. Clause 23: Ranges library

Change 23.6.3.3 [range.iota.iterator]:

namespace std::ranges {
  template<class W, class Bound>
  struct iota_view<W, Bound>::iterator {
  private:
    W value_ = W();             // exposition only
  public:
    [...]

    friend constexpr bool operator==(const iterator& x, const iterator& y)
      requires EqualityComparable<W>;
    friend constexpr bool operator!=(const iterator& x, const iterator& y)
      requires EqualityComparable<W>;

    friend constexpr bool operator<(const iterator& x, const iterator& y)
      requires StrictTotallyOrdered<W>;
    friend constexpr bool operator>(const iterator& x, const iterator& y)
      requires StrictTotallyOrdered<W>;
    friend constexpr bool operator<=(const iterator& x, const iterator& y)
      requires StrictTotallyOrdered<W>;
    friend constexpr bool operator>=(const iterator& x, const iterator& y)
      requires StrictTotallyOrdered<W>;
    friend constexpr compare_three_way_result_t<W> operator<=>(const iterator& x, const iterator& y)
      requires StrictTotallyOrdered<W> && ThreeWayComparable<W>

    [...]
  };
}

Change 23.6.3.3 [range.iota.iterator], paragraphs 14-20:

friend constexpr bool operator==(const iterator& x, const iterator& y)
  requires EqualityComparable<W>;
Effects: Equivalent to: return x.value_ == y.value_;
friend constexpr bool operator!=(const iterator& x, const iterator& y)
  requires EqualityComparable<W>;
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator<(const iterator& x, const iterator& y)
  requires StrictTotallyOrdered<W>;
Effects: Equivalent to: return x.value_ < y.value_;
[...]
friend constexpr bool operator>=(const iterator& x, const iterator& y)
  requires StrictTotallyOrdered<W>;
Effects: Equivalent to: return !(x < y);
friend constexpr compare_three_way_result_t<W> operator<=>(const iterator& x, const iterator& y)
  requires StrictTotallyOrdered<W> && ThreeWayComparable<W>;
Effects: Equivalent to: return x.value_ <=> y.value_;

Remove obsolete equality operators 23.6.3.4 [range.iota.sentinel]:

namespace std::ranges {
  template<class W, class Bound>
  struct iota_view<W, Bound>::sentinel {
  private:
    Bound bound_ = Bound();     // exposition only
  public:
    sentinel() = default;
    constexpr explicit sentinel(Bound bound);
    friend constexpr bool operator==(const iterator& x, const sentinel& y);
    friend constexpr bool operator==(const sentinel& x, const iterator& y);
    friend constexpr bool operator!=(const iterator& x, const sentinel& y);
    friend constexpr bool operator!=(const sentinel& x, const iterator& y);
  };
}

constexpr explicit sentinel(Bound bound);
Effects: Initializes bound_ with bound.
friend constexpr bool operator==(const iterator& x, const sentinel& y);
Effects: Equivalent to: return x.value_ == y.bound_;
friend constexpr bool operator==(const sentinel& x, const iterator& y);
Effects: Equivalent to: return y == x;
friend constexpr bool operator!=(const iterator& x, const sentinel& y);
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator& y);
Effects: Equivalent to: return !(y == x);

Remove operator!= from 23.7.4.3 [range.filter.iterator]:

namespace std::ranges {
  template<class V, class Pred>
  class filter_view<V, Pred>::iterator {
  private:
    iterator_t<V> current_ = iterator_t<V>();   // exposition only
    filter_view* parent_ = nullptr;             // exposition only
  public:
    [...]
    friend constexpr bool operator==(const iterator& x, const iterator& y)
      requires EqualityComparable<iterator_t<V>>;
    friend constexpr bool operator!=(const iterator& x, const iterator& y)
      requires EqualityComparable<iterator_t<V>>;
    [...]
  };
}

and

friend constexpr bool operator==(const iterator& x, const iterator& y)
  requires EqualityComparable<iterator_t<V>>;
Effects: Equivalent to: return x.current_ == y.current_;
friend constexpr bool operator!=(const iterator& x, const iterator& y)
  requires EqualityComparable<iterator_t<V>>;
Effects: Equivalent to: return !(x == y);

Remove obsolete equality operators from 23.7.4.4 [range.filter.sentinel]:

namespace std::ranges {
  template<class V, class Pred>
  class filter_view<V, Pred>::sentinel {
  private:
    sentinel_t<V> end_ = sentinel_t<V>();       // exposition only
  public:
    sentinel() = default;
    constexpr explicit sentinel(filter_view& parent);
    constexpr sentinel_t<V> base() const;
    friend constexpr bool operator==(const iterator& x, const sentinel& y);
    friend constexpr bool operator==(const sentinel& x, const iterator& y);
    friend constexpr bool operator!=(const iterator& x, const sentinel& y);
    friend constexpr bool operator!=(const sentinel& x, const iterator& y);
  };
}
constexpr explicit sentinel(filter_view& parent);
Effects: Initializes end_ with ranges::end(parent).
constexpr sentinel_t<V> base() const;
Effects: Equivalent to: return end_;
friend constexpr bool operator==(const iterator& x, const sentinel& y);
Effects: Equivalent to: return x.current_ == y.end_;
friend constexpr bool operator==(const sentinel& x, const iterator& y);
Effects: Equivalent to: return y == x;
friend constexpr bool operator!=(const iterator& x, const sentinel& y);
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator& y);
Effects: Equivalent to: return !(y == x);

Change 23.7.5.3 [range.transform.iterator]:

namespace std::ranges {
  template<class V, class F>
  template<bool Const>
  class transform_view<V, F>::iterator {
    [...]
    friend constexpr bool operator==(const iterator& x, const iterator& y)
      requires EqualityComparable<iterator_t<Base>>;
    friend constexpr bool operator!=(const iterator& x, const iterator& y)
      requires EqualityComparable<iterator_t<Base>>;

    friend constexpr bool operator<(const iterator& x, const iterator& y)
      requires RandomAccessRange<Base>;
    friend constexpr bool operator>(const iterator& x, const iterator& y)
      requires RandomAccessRange<Base>;
    friend constexpr bool operator<=(const iterator& x, const iterator& y)
      requires RandomAccessRange<Base>;
    friend constexpr bool operator>=(const iterator& x, const iterator& y)
      requires RandomAccessRange<Base>;
    friend constexpr compare_three_way_result_t<iterator_t<Base>> operator<=>(const iterator& x, const iterator& y)
      requires RandomAccessRange<Base>; && ThreeWayComparable<iterator_t<Base>>
    [...]
  };
}

Change 23.7.5.3 [range.transform.iterator], paragraphs 13-18:

friend constexpr bool operator==(const iterator& x, const iterator& y)
  requires EqualityComparable<iterator_t<Base>>;
Effects: Equivalent to: return x.current_ == y.current_;
friend constexpr bool operator!=(const iterator& x, const iterator& y)
  requires EqualityComparable<iterator_t<Base>>;
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator<(const iterator& x, const iterator& y)
  requires RandomAccessRange<Base>;
Effects: Equivalent to: return x.current_ < y.current_;
[...]
friend constexpr bool operator>=(const iterator& x, const iterator& y)
  requires RandomAccessRange<Base>;
Effects: Equivalent to: return !(x < y);
friend constexpr compare_three_way_result_t<iterator_t<Base>> operator<=>(const iterator& x, const iterator& y)
  requires RandomAccessRange<Base>; && ThreeWayComparable<iterator_t<Base>>
Effects: Equivalent to: return x.current_ <=> y.current_;

Remove obsolete equality operators from 23.7.5.4 [range.transform.sentinel]:

namespace std::ranges {
  template<class V, class F>
  template<bool Const>
  class transform_view<V, F>::sentinel {
    [...]
    friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
    friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);
    friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);
    friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);
    [...]
  };
}

and from 23.7.5.4 [range.transform.sentinel], paragraphs 4-7:

friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
Effects: Equivalent to: return x.current_ == y.end_;
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);
Effects: Equivalent to: return y == x;
friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);
Effects: Equivalent to: return !(y == x);

Remove obsolete equality operators from 23.7.6.3 [range.take.sentinel]:

namespace std::ranges {
  template<class V>
  template<bool Const>
  class take_view<V>::sentinel {
  private:
    using Base = conditional_t<Const, const V, V>;      // exposition only
    using CI = counted_iterator<iterator_t<Base>>;      // exposition only
    sentinel_t<Base> end_ = sentinel_t<Base>();         // exposition only
  public:
    [...]
    friend constexpr bool operator==(const sentinel& x, const CI& y);
    friend constexpr bool operator==(const CI& y, const sentinel& x);
    friend constexpr bool operator!=(const sentinel& x, const CI& y);
    friend constexpr bool operator!=(const CI& y, const sentinel& x);
  };
}

and from 23.7.6.3 [range.take.sentinel] paragraphs 4-5:

friend constexpr bool operator==(const sentinel& x, const CI& y);
friend constexpr bool operator==(const CI& y, const sentinel& x);
Effects: Equivalent to: return y.count() == 0 || y.base() == x.end_;
friend constexpr bool operator!=(const sentinel& x, const CI& y);
friend constexpr bool operator!=(const CI& y, const sentinel& x);
Effects: Equivalent to: return !(x == y);

Remove obsolete operator!= from 23.7.7.3 [range.join.iterator]:

namespace std::ranges {
template<class V>
  template<bool Const>
  struct join_view<V>::iterator {
    [...]
    friend constexpr bool operator==(const iterator& x, const iterator& y)
      requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> &&
               EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;

    friend constexpr bool operator!=(const iterator& x, const iterator& y)
      requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> &&
               EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;
    [...]
  };
}

and from 23.7.7.3 [range.join.iterator] paragraph 17:

friend constexpr bool operator==(const iterator& x, const iterator& y)
  requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> &&
           EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;
Effects: Equivalent to: return x.outer_ == y.outer_ && x.inner_ == y.inner_;
friend constexpr bool operator!=(const iterator& x, const iterator& y)
  requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> &&
           EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;
Effects: Equivalent to: return !(x == y);

Remove obsolete equality operators from 23.7.7.4 [range.join.sentinel]:

namespace std::ranges {
  template<class V>
  template<bool Const>
  struct join_view<V>::sentinel {
    [...]
    friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
    friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);
    friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);
    friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);
  };
}

and from 23.7.7.4 [range.join.sentinel] paragraphs 3-6:

friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
Effects: Equivalent to: return x.outer_ == y.end_;
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);
Effects: Equivalent to: return y == x;
friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);
Effects: Equivalent to: return !(y == x);

Remove obsolete equality operators from 23.7.8.3 [range.split.outer]:

namespace std::ranges {
  template<class V, class Pattern>
  template<bool Const>
  struct split_view<V, Pattern>::outer_iterator {
    [...]
    friend constexpr bool operator==(const outer_iterator& x, const outer_iterator& y)
      requires ForwardRange<Base>;
    friend constexpr bool operator!=(const outer_iterator& x, const outer_iterator& y)
      requires ForwardRange<Base>;

    friend constexpr bool operator==(const outer_iterator& x, default_sentinel_t);
    friend constexpr bool operator==(default_sentinel_t, const outer_iterator& x);
    friend constexpr bool operator!=(const outer_iterator& x, default_sentinel_t y);
    friend constexpr bool operator!=(default_sentinel_t y, const outer_iterator& x);
  };
}

and from 23.7.8.3 [range.split.outer] paragraphs 7-10:

friend constexpr bool operator==(const outer_iterator& x, const outer_iterator& y)
  requires ForwardRange<Base>;
Effects: Equivalent to: return x.current_ == y.current_;
friend constexpr bool operator!=(const outer_iterator& x, const outer_iterator& y)
  requires ForwardRange<Base>;
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator==(const outer_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const outer_iterator& x);
Effects: Equivalent to: return x.current_ == ranges::end(x.parent_->base_);
friend constexpr bool operator!=(const outer_iterator& x, default_sentinel_t y);
friend constexpr bool operator!=(default_sentinel_t y, const outer_iterator& x);
Effects: Equivalent to: return !(x == y);

Remove obsolete equality operators from 23.7.8.5 [range.split.inner]:

namespace std::ranges {
  template<class V, class Pattern>
  template<bool Const>
  struct split_view<V, Pattern>::inner_iterator {
    [...]
    friend constexpr bool operator==(const inner_iterator& x, const inner_iterator& y)
      requires ForwardRange<Base>;
    friend constexpr bool operator!=(const inner_iterator& x, const inner_iterator& y)
      requires ForwardRange<Base>;

    friend constexpr bool operator==(const inner_iterator& x, default_sentinel_t);
    friend constexpr bool operator==(default_sentinel_t, const inner_iterator& x);
    friend constexpr bool operator!=(const inner_iterator& x, default_sentinel_t y);
    friend constexpr bool operator!=(default_sentinel_t y, const inner_iterator& x);
    [...]
  };
}

and from 23.7.8.5 [ranges.split.inner] paragraphs 4-7:

friend constexpr bool operator==(const inner_iterator& x, const inner_iterator& y)
  requires ForwardRange<Base>;
Effects: Equivalent to: return x.i_.current_ == y.i_.current_;
friend constexpr bool operator!=(const inner_iterator& x, const inner_iterator& y)
  requires ForwardRange<Base>;
Effects: Equivalent to: return !(x == y);
friend constexpr bool operator==(const inner_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const inner_iterator& x);
Effects: Equivalent to:
auto cur = x.i_.current;
auto end = ranges::end(x.i_.parent_->base_);
if (cur == end) return true;
auto [pcur, pend] = subrange{x.i_.parent_->pattern_};
if (pcur == pend) return x.incremented_;
do {
  if (cur != pcur) return false;
  if (++pcur == pend) return true;
} while (++cur != end);
return false;

friend constexpr bool operator!=(const inner_iterator& x, default_sentinel_t y);
friend constexpr bool operator!=(default_sentinel_t y, const inner_iterator& x);
Effects: Equivalent to: return !(x == y);

4.10. Clause 24: Algorithms library

Change 24.4 [algorithm.syn]:

namespace std {
  [...]
  // [alg.3way], three-way comparison algorithms
  template<class T, class U>
    constexpr auto compare_3way(const T& a, const U& b);
  template<class InputIterator1, class InputIterator2, class Cmp>
    constexpr auto
      lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,
      lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1,
                                   InputIterator2 b2, InputIterator2 e2,
                                   Cmp comp)
        -> common_comparison_category_t<decltype(comp(*b1, *b2)), strong_ordering>;
  template<class InputIterator1, class InputIterator2>
    constexpr auto
      lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,
      lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1,
                                   InputIterator2 b2, InputIterator2 e2);
  [...]
}

Change 24.7.11 [alg.3way]:

template<class T, class U> constexpr auto compare_3way(const T& a, const U& b);

Effects: Compares two values and produces a result of the strongest applicable comparison category type:

Change 24.7.11 [alg.3way] paragraph 2:

template<class InputIterator1, class InputIterator2, class Cmp>
  constexpr auto
    lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,
    lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1,
                                 InputIterator2 b2, InputIterator2 e2,
                                 Cmp comp);

Change 24.7.11 [alg.3way] paragraph 4:

template<class InputIterator1, class InputIterator2>
  constexpr auto
    lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,
    lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1,
                                 InputIterator2 b2, InputIterator2 e2);
Effects: Equivalent to:
return lexicographical_compare_3way(b1, e1, b2, e2,
                                    [](const auto& t, const auto& u) {
                                      return compare_3way(t, u);
                                    });
return lexicographical_compare_three_way(b1, e1, b2, e2, compare_three_way());

4.11. Clause 25: Numeric library

No changes.

4.12. Clause 26: Time library

TBD

4.13. Clause 27: Localization library

TBD

4.14. Clause 28: Input/output library

TBD

4.15. Clause 29: Regular expressions library

TBD

4.16. Clause 30: Atomic operations library

TBD

4.17. Clause 31: Thread support library

TBD

5. References