Document number: P1825R0 Date: 2019-07-19 Project: Programming Language C++, Core Working Group Reply-to: David Stone: david@doublewise.net, david.stone@uber.com
P0527R1 ("Implicitly move from rvalue references in return statements" by David Stone) and P1155R3 ("More implicit moves" by Arthur O'Dwyer) both touch the same section of wording, including making some of the same changes. This wording accounts for all of the changes in both papers.
Change 11.9.5 [class.copy.elision]/3:
An implicitly movable entity is a variable of automatic storage duration that is either a non-volatile object or an rvalue reference to a non-volatile object type. In the following copy-initialization contexts, a move operation might be used instead of a copy operation:
If the expression in a
returnorco_returnstatement ([stmt.return]) is a (possibly parenthesized) id-expression that namesan object with automatic storage durationan implicitly movable entity declared in the body or parameter-declaration-clause of the innermost enclosing function or lambda-expression, or
if the operand of a throw-expression is
the name of a non-volatile automatic object (other than a function or catch-clause parameter)a (possibly parenthesized) id-expression that names an implicitly movable entity whose scope does not extend beyond theend of the innermost enclosing try-block (if there is one)compound-statement of the innermost try-block or function-try-block (if any) whose compound-statement or ctor-initializer encloses the throw-expression,overload resolution to select the constructor for the copy or the
[ Example:return_valueoverload to call is first performed as if theobject were designated byexpression or operand were an rvalue. If the first overload resolution fails or was not performed,or if the type of the first parameter of the selected constructor or theoverload resolution is performed again, considering thereturn_valueoverload is not an rvalue reference to the object's type (possibly cv-qualified),objectexpression or operand as an lvalue. [ Note: This two-stage overload resolution must be performed regardless of whether copy elision will occur. It determines the constructor or thereturn_valueoverload to be called if elision is not performed, and the selected constructor orthereturn_valueoverload must be accessible even if the call is elided. — end note ]-- end example ]void f() { T x; try { T y; try {g(x);} catch(...) { if(/*...*/) throw x; // does not move throw y; // moves } g(y); } catch(...) { g(x); // g(y); // error } }
Change Annex C.5: [diff.cpp17.class] to add an additional entry:
Affected subclauses: [class.copy.elision]
Change: A function returning an implicitly-movable entity may invoke a constructor taking an rvalue reference to a type different from that of the returned expression. Function and catch-clause parameters can be thrown using move constructors.
Rationale: Side-effect of making it easier to write more efficient code that takes advantage of moves.
Effect on original feature: Valid C++17 code may become ill-formed in this International Standard. For example:
struct base { base(); base(base const &); private: base(base &&); }; struct derived : base {}; base f(base b) { throw b; // error: base(base &&) is private derived d; return d; // error: base(base &&) is private }