1. Changelog
- 
     R2 (pre-Hagenberg): - 
       Forward to CWG (see § 5 Straw poll results). 
- 
       Fix one mistake in a drafting note ("deleted" to "ill-formed"). 
 
- 
       
- 
     R1: - 
       Add discussion of void operator = ( X && ) = default 
 
- 
       
2. Motivation
Current C++ permits 
| Today | After P2952 | 
|---|---|
| 
 | 
 | 
The comparison operators are inconsistent among themselves: 
| Today | After P2952 | 
|---|---|
| 
 | 
 | 
The status quo is inconsistent between non-defaulted and defaulted functions,
making it unnecessarily tedious to upgrade to 
| Today | After P2952 | 
|---|---|
| 
 | 
 | 
| 
 | 
 | 
The ill-formedness of these declarations comes from overly restrictive wording in the standard,
such as [class.eq]/1 specifically requiring that a defaulted equality
operator must have a declared return type of 
This proposal does not seek to change the set of valid return types for these functions.
We propose a purely syntactic change to expand the range of allowed declaration syntax, not semantics.
(But we do one drive-by clarification which we believe matches EWG’s original intent:
if an empty class’s defaulted 
3. Proposal
We propose that a defaulted function declaration with a placeholder return type should have its type deduced ([dcl.spec.auto.general]) as if from a fictional return statement that returns:
- 
     a prvalue of type bool operator == operator != operator < operator > operator <= operator >= 
- 
     a prvalue of type Q operator <=> Q 
- 
     an lvalue of type C operator = C 
Then, the deduced return type is compared to the return type(s) permitted by the standard. If the types match, the declaration is well-formed. Otherwise it’s ill-formed.
struct MyClass { auto & operator = ( const MyClass & ) = default ; // Proposed OK: deduces MyClass& decltype ( auto ) operator = ( const MyClass & ) = default ; // Proposed OK: deduces MyClass& auto && operator = ( const MyClass & ) = default ; // Proposed OK: deduces MyClass& const auto & operator = ( const MyClass & ) = default ; // Still ill-formed: deduced const MyClass& is not MyClass& auto operator = ( const MyClass & ) = default ; // Still ill-formed: deduced MyClass is not MyClass& auto * operator = ( const MyClass & ) = default ; // Still ill-formed: deduction fails void operator = ( const MyClass & ) = default ; // Still ill-formed: void is not MyClass& }; 
For 
struct MyClass { auto operator == ( const MyClass & ) const = default ; // Proposed OK: deduces bool decltype ( auto ) operator == ( const MyClass & ) const = default ; // Proposed OK: deduces bool auto && operator == ( const MyClass & ) const = default ; // Still ill-formed: deduced bool&& is not bool auto & operator == ( const MyClass & ) const = default ; // Still ill-formed: deduction fails }; 
3.1. "Return type" versus "declared return type"
Today, vendors unanimously reject 
We tentatively propose to leave [dcl.fct.def.default] alone, and simply add an example that indicates the (new) intent of the (existing) wording: that it should now be interpreted as talking about the assignment operator’s actual return type, not its declared (placeholder) return type.
3.2. "Defaulted as deleted"
The current wording for comparison operators is crafted so that the following 
template < class T > struct Container { T t ; auto operator <=> ( const Container & ) const = default ; }; Container < std :: mutex > cm ; // OK, <=> is deleted struct Weird { int operator <=> ( Weird ) const ; }; Container < Weird > cw ; // OK, <=> is deleted because Weird's operator<=> // returns a non-comparison-category type 
Similarly for dependent return types:
template < class R > struct C { int i ; R operator <=> ( const C & ) const = default ; }; static_assert ( std :: three_way_comparable < C < std :: strong_ordering >> ); static_assert ( ! std :: three_way_comparable < C < int >> ); // OK, C<int>'s operator<=> is deleted 
Therefore we can’t just say "
3.3. "Deducing this 
   [CWG2586] (adopted for C++23) permits defaulted functions to have explicit object parameters.
This constrains the wordings we can choose for 
There’s a quirk with rvalue-ref-qualified assignment operators — not move assignment, but assignment where the destination object is explicitly rvalue-ref-qualified.
- 
     auto && operator = ( const B & ) && { return * this ; } 
- 
     auto && operator = ( this B && self , const B & ) { return self ; } 
Nonetheless, a defaulted assignment operator always returns an lvalue reference ([class.copy.assign]/6, [dcl.fct.def.default]/2.5), regardless of whether it’s declared using explicit object syntax.
struct A { A & operator = ( const A & ) && = default ; // OK today A && operator = ( const A & ) && = default ; // Ill-formed, return type isn't A& decltype ( auto ) operator = ( const A & ) && { return * this ; } // OK, deduces A& decltype ( auto ) operator = ( const A & ) && = default ; // Proposed OK, deduces A& }; struct B { B & operator = ( this B && self , const B & ) { return self ; } // Error, self can't bind to B& B && operator = ( this B && self , const B & ) { return self ; } // OK decltype ( auto ) operator = ( this B && self , const B & ) { return self ; } // OK, deduces B&& B & operator = ( this B & self , const B & ) = default ; // OK B & operator = ( this B && self , const B & ) = default ; // OK B && operator = ( this B && self , const B & ) = default ; // Ill-formed, return type isn't B& decltype ( auto ) operator = ( this B && self , const B & ) = default ; // Proposed OK, deduces B& }; 
Defaulted rvalue-ref-qualified assignment operators are weird; Arthur is bringing another paper to forbid them entirely ([P2953]). However, P2952 doesn’t need to treat them specially. Defaulted assignment operators invariably return lvalue references, so we invariably deduce as-if-from an lvalue reference, full stop.
3.4. Burden on specifying new defaultable operators
We propose to leave [dcl.fct.def.default] alone and reinterpret its term "return type" to mean the actual return type, not the declared return type. This will, by default, permit the programmer to use placeholder return types on their defaulted operators. So there is a new burden on the specification of the defaultable operator, to specify exactly how return type deduction works for the implicitly defined operator.
operator ++ ( int ) A defaulted postfixfor classoperator ++ shall have a return type that isX orX . If its declared return type contains a placeholder type, its return type is deduced as if fromvoid 
wherereturn X ( r ); is an lvalue reference to the function’s object parameter, ifr is a well-formed expression;X ( r ) 
otherwise.return ; 
[P0847] §5.2’s
example of 
struct add_postfix_increment { template < class Self > auto operator ++ ( this Self & , int ) = default ; // Today: ill-formed, can't default operator++ // After P1046R2: presumably still not OK, can't default a template }; struct S : add_postfix_increment { int i ; auto & operator ++ () { ++ i ; return * this ; } using add_postfix_increment :: operator ++ ; }; S s = { 1 }; S t = s ++ ; 
3.5. Other return types are still forbidden
Notice that returning any type other than 
struct X { void operator = ( X && ) = default ; // still ill-formed }; 
Any paper that did propose to permit defaulting 
But none of this is a problem for this proposal P2952, because we propose to continue rejecting
a defaulted 
3.6. Existing corner cases
There is vendor divergence in some corner cases. Here is a table of the divergences we found, plus our opinion as to the conforming behavior, and our proposed behavior.
| URL | Code | Clang | GCC | MSVC | EDG | Correct | 
|---|---|---|---|---|---|---|
| link | 
 | ✗ | ✗ | ✗ | ✓ | ✗ | 
| link | 
 | ✓ | ✗ | ✓ | ✓ | ✓ | 
| link | 
 | ✗ | ✓ | ✗ | ✗ | Today: ✗ Proposed: ✓ | 
| link | 
 | ✗ | ✗ | ✓ | ✓ | Today: ✗ Proposed: ✗ | 
| link | 
 | ✗ | ✓ | ✓ | ✓ | Today: ✗ Proposed: ✓ | 
| link | 
 | ✗ | unmet | ✓ | ✓ | Today: ✗ Proposed: unmet | 
| link | 
 | ✓ | ✓ | ✗ | deleted | Today: ✓ Proposed: deleted | 
| link | 
 | ✓ | ✓ | ✗ | deleted | deleted | 
| link | 
 | ✓ | deleted | ✓ | deleted | deleted | 
| link | 
 | ✓ | ✓ | ✓ | deleted | Today: ✓ Proposed: deleted | 
| link | 
 | ✓ | ✗ | ✓ | deleted | deleted | 
| link | 
 | ✓ | ✓ | ✓ | ✓ | deleted | 
| link | 
 | ✓ | ✓ | noexcept | incon- sistent | ✓ | 
| link | 
 | ✓ | ✓ | ✓ | ✓ | ✓ | 
| link | 
 | deleted | ✗ | ✗ | deleted | deleted | 
| link | 
 | deleted | ✗ | ✗ | deleted | deleted | 
| link | 
 | ✓ | ✓ | ✓ | ✓ | ✓ ([P2953]: deleted) | 
| link | 
 | ✗ | ✗ | ✗ | ✗ | ✗ | 
3.7. Impact on existing code
There should be little effect on existing code, since this proposal mainly allows syntax that was ill-formed before. As shown in § 3.6 Existing corner cases, we do propose to change some very arcane examples, e.g.
struct C { const std :: strong_ordering & operator <=> ( const C & ) const = default ; // Today: Well-formed, non-deleted // Tomorrow: Well-formed, deleted }; 
4. Implementation experience
None yet.
5. Straw poll results
Arthur O’Dwyer presented P2592R1 in the EWG telecon of 2025-01-08. The following straw polls were taken. The first was interpreted as "no consensus"; the second was interpreted as consensus (pending electronic polling).
| SF | F | N | A | SA | |
|---|---|---|---|---|---|
| EWG prefers this paper contains the change in P2953 (banning explicitly defaulted operator= with rvalue ref-qualifier). [Chair: This means EWG wants to see this paper again.] | 2 | 4 | 9 | 1 | 0 | 
| Forward P2952R1 to CWG for inclusion in C++26, pending online polling. | 3 | 10 | 4 | 1 | 0 | 
6. Proposed wording
6.1. [class.eq]
DRAFTING NOTE: The phrase "equality operator function" ([over.binary])
means 
Modify [class.eq] as follows:
1․ A defaulted equality operator function ([over.binary]) shall have a declared return type.bool 2․ A defaulted
operator function for a class== is defined as deleted unless, for eachC i in the expanded list of subobjects for an objectx of typex ,C ix i is usable ([class.compare.default]).== x 3․ The return value
x․ A defaultedof a defaultedV operator function with parameters== andx is determined by comparing corresponding elementsy i andx i in the expanded lists of subobjects fory andx (in increasing index order) until the first index i wherey ix == iy yields a result value which,whencontextually converted to, yieldsbool false. If no such index exists,isV true. Otherwise,isV false.operator function shall have the return type== . If its declared return type contains a placeholder type, its return type is deduced as if frombool .return true; 4․ [Example 1:
struct D { int i ; friend bool operator == ( const D & x , const D & y ) = default ; // OK, returns x.i == y.i }; — end example]
6.2. [class.spaceship]
DRAFTING NOTE: There are only three "comparison category types" in C++, and 
Modify [class.spaceship] as follows:
[...]2․ Let
be the declared return type of a defaulted three-way comparison operator function, and letR i be the elements of the expanded list of subobjects for an objectx of typex .C — (2.1) If
R iscontains a placeholder type, then let cvi,auto i be the type of the expressionR ix i. The operator function is defined as deleted if that expression is not usable or if<=> x i is not a comparison category type ([cmp.categories.pre]) for any i. The return type is deduced as if fromR , wherereturn Q ( std :: strong_ordering :: equal ); is the common comparison type (see below) ofQ 0,R 1, ...,R n-1.R — (2.2) Otherwise,
if the synthesized three-way comparison of typeshall not contain a placeholder type. IfR between any objectsR i andx i is not defined, the operator function is defined as deleted.x 3․ The return value
x․ A defaulted three-way comparison operator function which is not deleted shall have a return type which is a comparison category type ([cmp.categories.pre]).V of typeof the defaulted three-way comparison operator function with parametersR andx y of the same typeis determined by comparing corresponding elementsi andx i in the expanded lists of subobjects fory andx (in increasing index order) until the first index i where the synthesized three-way comparison of typey betweenR i andx i yields a result valuey i wherev iv , contextually converted to!= 0 , yieldsbool true;isV a copy ofi. If no such index exists,v isV static_cast < R > ( std :: strong_ordering :: equal .) 4․ The common comparison type
of a possibly-empty list of n comparison category typesU 0,T 1, ...,T n-1 is defined as follows:T [...]
6.3. [class.compare.secondary]
Modify [class.compare.secondary] as follows:
1․ A secondary comparison operator is a relational operator ([expr.rel]) or the
operator.!= A defaulted operator function ([over.binary]) for a secondary comparison operator@shall have a declared return type.bool 2․
TheA defaulted secondary comparison operator function with parametersandx is defined as deleted ify — (2.1) overload resolution ([over.match]), as applied to
, does not result in a usable candidate, orx @y — (2.2) the candidate selected by overload resolution is not a rewritten candidate.
Otherwise, the operator function yields
x․ A defaulted secondary comparison operator function shall have the return type. The defaulted operator function is not considered as a candidate in the overload resolution for thex @y @operator.. If its declared return type contains a placeholder type, its return type is deduced as if frombool .return true; 3․ [Example 1:
struct HasNoLessThan { }; struct C { friend HasNoLessThan operator <=> ( const C & , const C & ); bool operator < ( const C & ) const = default ; // OK, function is deleted }; — end example]
6.4. [class.copy.assign]
DRAFTING NOTE: [class.copy.assign]/6 already clearly states that "The implicitly-declared copy/move assignment operator for class 
Modify [class.copy.assign] as follows:
14․ The implicitly-defined copy/move assignment operator for a class returns the object for which the assignment operator is invoked, that is, the object assigned to.15․ If a defaulted copy/move assignment operator’s declared return type contains a placeholder type, its return type is deduced as if from
, wherereturn r ; is an lvalue reference to the object for which the assignment operator is invoked.r 16․ [Example:
—end example]struct A { decltype ( auto ) operator = ( A && ) = default ; // Return type is A& }; struct B { auto operator = ( B && ) = default ; // error: Return type is B, which violates [dcl.fct.def.default]/2.5 };