1. Abstract
We propose deprecating most of . See §3 Wording for the details.
The proposed deprecation preserves the useful parts of , and removes
the dubious / already broken ones. This paper aims at breaking at compile-time
code which is today subtly broken at runtime or through a compiler update. The
paper might also break another type of code: that which doesn’t exist. This
removes a significant foot-gun and removes unintuitive corner cases from the
languages.
The first version of this paper, [P1152R0], has extensive background information which is not repeated here:
See [P1382R0] for the follow-up paper on / requested by SG1.
2. Edit History
2.1. r2 → r3
[P1152R2] was reviewed offline by Alisdair Meredith.
-
should have beenis_volatile < T > .is_volatile_v < T > -
Use Mandates instead of Constraints to allow marking a function as deleted or employing a
instead of usingstatic_assert orenable_if . This likely provides better error messages.requires -
"
isis_volatile_v < T > false" was erroneously used instead of testing whether thepointer itself wasthis . Wording of the Mandates clause was updated to only apply to thevolatile overloads of methods instead.volatile
2.2. r1 → r2
[P1152R1] was seen by SG1 and EWG in Kona. This update does the following:
-
Also edit sections [expr.post.incr], [expr.pre.incr], [expr.ass], which are redundant with other sections already modified by this paper.
-
Change Remarks to Constraints per [P1369R0].
-
Change language wording to explicitly call out deprecation.
| Poll | Group | SF | F | N | A | SA | Outcome |
|---|---|---|---|---|---|---|---|
| Forward this paper—with edits as discussed—to EWG for C++20. | SG1 | 3 | 12 | 1 | 1 | 0 | ✅ |
| Proposal as presented for C++20. | EWG | 6 | 27 | 1 | 0 | 0 | ✅ |
2.3. r0 → r1
[P1152R0] was seen by SG1 and EWG in San Diego. This update does the following:
-
Remove background information from the paper.
-
Follow the guidance from SG1 and EWG, based on the polls below.
| Poll | Group | SF | F | N | A | SA | Outcome |
|---|---|---|---|---|---|---|---|
Deprecate compound operations (including and ) on scalar types (arithmetic, pointer, enumeration).
| SG1 | 4 | 19 | 3 | 0 | 0 | ✅ |
Deprecate compound operations (including and ) on scalar types (arithmetic, pointer, enumeration).
| EWG | 4 | 9 | 4 | 0 | 0 | ✅ |
Deprecate usage of assignment chaining on scalar types (arithmetic, pointer, enumeration, pointer to members, ).
| SG1 | 6 | 15 | 3 | 0 | 0 | ✅ |
Deprecate usage of assignment chaining on scalar types (arithmetic, pointer, enumeration, pointer to members, ).
| EWG | 6 | 9 | 3 | 0 | 0 | ✅ |
SG1 would be OK if we deprecated qualified member functions (pending separate decision on what we do with atomic).
| SG1 | 1 | 5 | 10 | 4 | 3 | ❌ |
EWG would be OK if we deprecated qualified member functions (pending separate decision on what we do with atomic).
| EWG | 2 | 7 | 7 | 1 | 0 | ✅ |
SG1 would be OK if we deprecated partial template specializations, overloads, or qualified member functions in the STL for all but the atomic, , and type traits (, , etc) parts of the Library.
| SG1 | 1 | 9 | 6 | 2 | 0 | ✅ |
EWG would be OK if we deprecated partial template specializations, overloads, or qualified member functions in the STL for all but the atomic, , and type traits (, , etc) parts of the Library.
| EWG | 1 | 11 | 9 | 0 | 0 | ✅ |
Deprecate member functions of atomic in favor of new template partial specializations which will only declare load, store, and only exist when is true.
| SG1 | 2 | 1 | 1 | 11 | 2 | ❌ |
Deprecate member functions of atomic in favor of new template partial specializations which will only declare load, store, RMW, and only exist when is true.
| SG1 | 4 | 7 | 3 | 3 | 0 | ✅ |
Deprecate member functions of atomic in favor of new template partial specializations which will only declare load, store, RMW, and only exist when is true.
| EWG | 2 | 9 | 3 | 0 | 0 | ✅ |
Deprecate member functions of atomic in favor of new template partial specializations which will only declare load, store, RMW.
| SG1 | 0 | 0 | 0 | 10 | 7 | ❌ |
SG1 would be OK if we deprecated top-level parameters.
| SG1 | 6 | 9 | 6 | 2 | 1 | ✅ |
EWG would be OK if we deprecated top-level parameters.
| EWG | 6 | 9 | 6 | 0 | 0 | ✅ |
| EWG would be OK if we deprecated top-level const parameters. | EWG | 0 | 2 | 5 | 8 | 8 | ❌ |
SG1 would be OK if we deprecated top-level return values.
| SG1 | 6 | 9 | 4 | 2 | 0 | ✅ |
EWG would be OK if we deprecated top-level return values.
| EWG | 6 | 6 | 5 | 0 | 0 | ✅ |
| EWG would be OK if we deprecated top-level const return values. | EWG | 2 | 3 | 3 | 5 | 5 | ❌ |
SG1 interested is interested in hearing about / free functions in a separate paper, given that time is limited and we could be doing something else.
| SG1 | 0 | 17 | 4 | 3 | 0 | ✅ |
EWG interested is interested in hearing about / free functions in a separate paper, given that time is limited and we could be doing something else.
| EWG | 2 | 11 | 4 | 1 | 0 | ✅ |
3. Wording
The proposed wording follows the language and library approach to deprecation:
-
Language deprecation is called out in the Standard text itself, and repeated in Annex D.
-
Library deprecation presents the library without the deprecated feature, and only mentions said feature in Annex D.
3.1. Program execution [intro.execution]
No changes.
Accesses through
glvalues are evaluated strictly according to the rules of the abstract machine.volatile Reading an object designated by a
glvalue, modifying an object, calling a library I/O function, or calling a function that does any of those operations are all side effects, which are changes in the state of the execution environment. Evaluation of an expression (or a subexpression) in general includes both value computations (including determining the identity of an object for glvalue evaluation and fetching a value previously assigned to an object for prvalue evaluation) and initiation of side effects. When a call to a library I/O function returns or an access through avolatile glvalue is evaluated the side effect is considered complete, even though some external actions implied by the call (such as the I/O itself) or by thevolatile access may not have completed yet.volatile
3.2. Data races [intro.races]
No changes.
Two accesses to the same object of type
volatile do not result in a data race if both occur in the same thread, even if one or more occurs in a signal handler. For each signal handler invocation, evaluations performed by the thread invoking a signal handler can be divided into two groups A and B, such that no evaluations in B happen before evaluations in A, and the evaluations of suchstd :: sig_atomic_t volatile objects take values as though all evaluations in A happened before the execution of the signal handler and the execution of the signal handler happened before all evaluations in B.std :: sig_atomic_t
3.3. Forward progress [intro.progress]
No changes.
The implementation may assume that any thread will eventually do one of the following:
terminate,
make a call to a library I/O function,
perform an access through a
glvalue, orvolatile perform a synchronization operation or an atomic operation
During the execution of a thread of execution, each of the following is termed an execution step:
termination of the thread of execution,
performing an access through a
glvalue, orvolatile completion of a call to a library I/O function, a synchronization operation, or an atomic operation.
3.4. Increment and decrement [expr.post.incr]
Modify as follows.
The value of a postfix
expression is the value of its operand. [ Note: The value obtained is a copy of the original value —end note ] The operand shall be a modifiable lvalue. The type of the operand shall be++ an arithmetic type other than cva non-bool qualified arithmetic type other thanvolatile , or a non-bool qualified pointer to a complete object type. The type of the operand can be avolatile -qualified arithmetic type other than cvvolatile , or abool -qualified pointer to a complete object type (this usage is deprecated; see [depr.volatile]). The value of the operand object is modified by addingvolatile to it. The value computation of the1 expression is sequenced before the modification of the operand object. With respect to an indeterminately-sequenced function call, the operation of postfix++ is a single evaluation. [ Note: Therefore, a function call shall not intervene between the lvalue-to-rvalue conversion and the side effect associated with any single postfix++ operator. —end note ] The result is a prvalue. The type of the result is the cv-unqualified version of the type of the operand. If the operand is a bit-field that cannot represent the incremented value, the resulting value of the bit-field is implementation-defined. See also [expr.add] and [expr.ass].++ The operand of postfix
is decremented analogously to the postfix-- operator. [ Note: For prefix increment and decrement, see [expr.pre.incr]. —end note ]++
3.5. Class member access [expr.ref]
No changes.
Abbreviating postfix-expression.id-expression as
,E1 . E2 is called the object expression. IfE1 is a bit-field,E2 is a bit-field. The type and value category ofE1 . E2 are determined as follows. In the remainder of [expr.ref], cq represents eitherE1 . E2 or the absence ofconst and vq represents eitherconst or the absence ofvolatile . cv represents an arbitrary set of cv-qualifiers.volatile
If
is a non-static data member and the type ofE2 is “cq1 vq1 X”, and the type ofE1 is “cq2 vq2 T”, the expression designates the named member of the object designated by the first expression. IfE2 is an lvalue, thenE1 is an lvalue; otherwiseE1 . E2 is an xvalue. Let the notation vq12 stand for the “union” of vq1 and vq2; that is, if vq1 or vq2 isE1 . E2 , then vq12 isvolatile . Similarly, let the notation cq12 stand for the “union” of cq1 and cq2; that is, if cq1 or cq2 isvolatile , then cq12 isconst . Ifconst is declared to be aE2 member, then the type ofmutable is “vq12 T”. IfE1 . E2 is not declared to be aE2 member, then the type ofmutable is “cq12 vq12 T”.E1 . E2
3.6. Increment and decrement [expr.pre.incr]
Modify as follows.
The operand of prefix
is modified by adding++ . The operand shall be a modifiable lvalue. The type of the operand shall be1 an arithmetic type other than cva non-bool -qualified arithmetic type other thanvolatile , or a non-bool -qualified pointer to a completely-defined object type. The type of the operand can be avolatile -qualified arithmetic type other than cvvolatile , or abool -qualified pointer to a completely-defined object type (this usage is deprecated; see [depr.volatile]). The result is the updated operand; it is an lvalue, and it is a bit-field if the operand is a bit-field. The expressionvolatile is equivalent to++ x . [ Note: See the discussions of [expr.add] and assignment operators [expr.ass] for information on conversions. —end note ]x += 1 The operand of prefix
is modified by subtracting-- . The requirements on the operand of prefix1 and the properties of its result are otherwise the same as those of prefix-- . [ Note: For postfix increment and decrement, see [expr.post.incr]. —end note ]++
3.7. Assignment and compound assignment operators [expr.ass]
Modify as follows.
- The assignment operator (
) and the compound assignment operators all group right-to-left.= - All require a modifiable lvalue as their left operand; their result is an lvalue referring to the left operand. The result in all cases is a bit-field if the left operand is a bit-field. In all cases, the assignment is sequenced after the value computation of the right and left operands, and before the value computation of the assignment expression. The right operand is sequenced before the left operand. With respect to an indeterminately-sequenced function call, the operation of a compound assignment is a single evaluation. [ Note: Therefore, a function call shall not intervene between the lvalue-to-rvalue conversion and the side effect associated with any single compound assignment operator. —end note ]
assignment - expression conditional - expression logical - or - expression assignment - operator initializer - clause throw - expression assignment - operator : one of = *= /= %= += -= >>= <<= &= ^= |= - In simple assignment (
), the object referred to by the left operand is modified by replacing its value with the result of the right operand.= - If the left operand is not of class type and is not
, the expression is implicitly converted to the cv-unqualified type of the left operand.volatile - If the left operand is not of class type and is
, the expression is implicitly converted to the cv-unqualified type of the left operand (this usage is deprecated; see [depr.volatile]).volatile - If the left operand is of class type, the class shall be complete. Assignment to objects of a class is defined by the copy/move assignment.
- [ Note: For class objects, assignment is not in general the same as initialization. —end note ]
- When the left operand of an assignment operator is a bit-field that cannot represent the value of the expression, the resulting value of the bit-field is implementation-defined.
- Expressions of the form
whereE1 op = E2 is aE1 -qualified arithmetic or pointer type shall either be a discarded-value expression or appear in an unevaluated context (other usages are deprecated; see [depr.volatile]).volatile TheOtherwise, the behavior of an expression of the formis equivalent toE1 op = E2 except thatE1 = E1 op E2 is evaluated only once. InE1 and+= ,-= shall either have arithmetic type or be a pointer to a possibly cv-qualified completely-defined object type. In all other cases,E1 shall have arithmetic type.E1 - If the value being stored in an object is read via another object that overlaps in any way the storage of the first object, then the overlap shall be exact and the two objects shall have the same type, otherwise the behavior is undefined. [ Note: This restriction applies to the relationship between the left and right sides of the assignment operation; it is not a statement about how the target of the assignment may be aliased in general. See [basic.lval]. —end note ]
- A braced-init-list may appear on the right-hand side of
- an assignment to a scalar, in which case the initializer list shall have at most a single element. The meaning of
, wherex = { v } is the scalar type of the expressionT , is that ofx . The meaning ofx = T { v } isx = {} .x = T {} - an assignment to an object of class type, in which case the initializer list is passed as the argument to the assignment operator function selected by overload resolution.
3.8. The cv-qualifiers [dcl.type.cv]
No changes.
The semantics of an access through a
glvalue are implementation-defined. If an attempt is made to access an object defined with avolatile -qualified type through the use of a non-volatile glvalue, the behavior is undefined.volatile [ Note:
is a hint to the implementation to avoid aggressive optimization involving the object because the value of the object might be changed by means undetectable by an implementation. Furthermore, for some implementations,volatile might indicate that special hardware instructions are required to access the object. See [intro.execution] for detailed semantics. In general, the semantics ofvolatile are intended to be the same in C++ as they are in C. —end note ]volatile
3.9. Functions [dcl.fct]
Modify as follows.
The parameter-declaration-clause determines the arguments that can be specified, and their processing, when the function is called. [ Note: The parameter-declaration-clause is used to convert the arguments specified on the function call; see [expr.call] —end note ] If the parameter-declaration-clause is empty, the function takes no arguments. A parameter list consisting of a single unnamed parameter of non-dependent type
is equivalent to an empty parameter list. Except for this special case, a parameter shall not have type cvvoid . A parameter can have avoid -qualified type (this usage is deprecated; see [depr.volatile]). If the parameter-declaration-clause terminates with an ellipsis or a function parameter pack, the number of arguments shall be equal to or greater than the number of parameters that do not have a default argument and are not function parameter packs. Where syntactically correct and where "volatile " is not part of an abstract-declarator, "... " is synonymous with ", ... ".... [...]
The type of a function is determined using the following rules. The type of each parameter (including function parameter packs) is determined from its own decl-specifier-seq and declarator. After determining the type of each parameter, any parameter of type "array of
" or of function typeT is adjusted to be "pointer toT ". After producing the list of parameter types, any top-level cv-qualifiers modifying a parameter type are deleted when forming the function type. The resulting list of transformed parameter types and the presence or absence of the ellipsis or a function parameter pack is the function’s parameter-type-list.T [...]
Functions shall not have a return type of type array or function, although they may have a return type of type pointer or reference to such things. There shall be no arrays of functions, although there can be arrays of pointers to functions.
Functions can have aqualified return type (this usage is deprecated; see [depr.volatile]).volatile
3.10. Non-static member functions [class.mfct.non-static]
No changes.
A non-static member function may be declared
,const , orvolatile . These cv-qualifiers affect the type of theconst volatile pointer. They also affect the function type of the member function; a member function declaredthis is aconst member function, a member function declaredconst is avolatile member function and a member function declaredvolatile is aconst volatile member function.const volatile
3.11. The this pointer [class.this]
No changes.
In the body of a non-static member function, the keyword
is a prvalue expression whose value is the address of the object for which the function is called. The type ofthis in a member function of a classthis isX . If the member function is declaredX * , the type ofconst isthis , if the member function is declaredconst X * , the type ofvolatile isthis , and if the member function is declaredvolatile X * , the type ofconst volatile isthis .const volatile X *
semantics apply involatile member functions when accessing the object and its non-static data members.volatile
3.12. Constructors [class.ctor]
No changes.
A constructor can be invoked for a
,const orvolatile object.const volatile andconst semantics are not applied on an object under construction. They come into effect when the constructor for the most derived object ends.volatile
3.13. Destructors [class.dtor]
No changes.
A destructor is used to destroy objects of its class type. The address of a destructor shall not be taken. A destructor can be invoked for a
,const orvolatile object.const volatile andconst semantics are not applied on an object under destruction. They stop being in effect when the destructor for the most derived object starts.volatile
3.14. Overloadable declarations [over.load]
Modify as follows.
Parameter declarations that differ only in the presence or absence of
Parameter declarations that differ only in the presence or absence ofconst and/orare equivalent. That is, thevolatile const andtype-specifier for each parameter typetype-specifiersvolatile areis ignored when determining which function is being declared, defined, or called.are equivalent (this usage is deprecated; see [depr.volatile]).volatile
3.15. Built-in operators [over.built]
Modify as follows.
In the remainder of this section, vq represents either
or no cv-qualifier (usage of vq representingvolatile is deprecated; see [depr.volatile]) .volatile For every pair (T, vq), where T is an arithmetic type other than
, there exist candidate operator functions of the formbool vq T & operator ++ ( vq T & ); T operator ++ ( vq T & , int ); For every pair (T, vq), where T is an arithmetic type other than
, there exist candidate operator functions of the formbool vq T & operator -- ( vq T & ); T operator -- ( vq T & , int ); For every pair (T, vq), where T is a cv-qualified or cv-unqualified object type, there exist candidate operator functions of the form
T * vq & operator ++ ( T * vq & ); T * vq & operator -- ( T * vq & ); T * operator ++ ( T * vq & , int ); T * operator -- ( T * vq & , int ); For every quintuple (C1, C2, T, cv1, cv2), where C2 is a class type, C1 is the same type as C2 or is a derived class of C2, and T is an object type or a function type, there exist candidate operator functions of the form
cv12 T & operator ->* ( cv1 C1 * , cv2 T C2 ::* ); For every triple (L, vq, R), where L is an arithmetic type, and R is a promoted arithmetic type, there exist candidate operator functions of the form
vq L & operator = ( vq L & , R ); vq L & operator *= ( vq L & , R ); vq L & operator /= ( vq L & , R ); vq L & operator += ( vq L & , R ); vq L & operator -= ( vq L & , R ); For every pair (T, vq), where T is any type, there exist candidate operator functions of the form
T * vq & operator = ( T * vq & , T * ); For every pair (T, vq), where T is an enumeration or pointer to member type, there exist candidate operator functions of the form
vq T & operator = ( vq T & , T ); For every pair (T, vq), where T is a cv-qualified or cv-unqualified object type, there exist candidate operator functions of the form
T * vq & operator += ( T * vq & , std :: ptrdiff_t ); T * vq & operator -= ( T * vq & , std :: ptrdiff_t ); For every triple (L, vq, R), where L is an integral type, and R is a promoted integral type, there exist candidate operator functions of the form
vq L & operator %= ( vq L & , R ); vq L & operator <<= ( vq L & , R ); vq L & operator >>= ( vq L & , R ); vq L & operator &= ( vq L & , R ); vq L & operator ^= ( vq L & , R ); vq L & operator |= ( vq L & , R );
3.16. Tuples [tuple]
Modify as follows.
Header
synopsis [tuple.syn]:< tuple > namespace std { [...] // [ tuple.helper , tuple helper classes template < class T > class tuple_size ; // not defined template < class T > class tuple_size < const T > ; template < class T > class tuple_size < volatile T > ; template < class T > class tuple_size < const volatile T > ; template < class ... Types > class tuple_size < tuple < Types ... >> ; template < size_t I , class T > class tuple_element ; // not defined template < size_t I , class T > class tuple_element < I , const T > ; template < size_t I , class T > class tuple_element < I , volatile T > ; template < size_t I , class T > class tuple_element < I , const volatile T > ; [...] } [...]
Tuple helper classes [tuple.helper]
template < class T > class tuple_size < const T > ; template < class T > class tuple_size < volatile T > ; template < class T > class tuple_size < const volatile T > ; Let
denoteTS of the cv-unqualified typetuple_size < T > . If the expressionT is well-formed when treated as an unevaluated operand, then each of the three templates shall satisfy theTS :: value requirements with a base characteristic ofTransformationTrait integral_constant < size_t , TS :: value > Otherwise, they shall have no member
.value Access checking is performed as if in a context unrelated to
andTS . Only the validity of the immediate context of the expression is considered. [ Note: The compilation of the expression can result in side effects such as the instantiation of class template specializations and function template specializations, the generation of implicitly-defined functions, and so on. Such side effects are not in the "immediate context" and can result in the program being ill-formed. —end note ]T In addition to being available via inclusion of the
header, the< tuple > three templates aretemplate is available when any of the headers,< array > , or< ranges > are included.< utility > template < size_t I , class T > class tuple_element < I , const T > ; template < size_t I , class T > class tuple_element < I , volatile T > ; template < size_t I , class T > class tuple_element < I , const volatile T > ; Let
denoteTE of the cv-unqualified typetuple_element_t < I , T > . ThenT each of the three templatesthe template shall satisfy therequirements with a member typedefTransformationTrait that names the following type:type .add_const_t < TE >
- for the first specialization,
,add_const_t < TE > - for the second specialization,
, andadd_volatile_t < TE > - for the third specialization,
.add_cv_t < TE > In addition to being available via inclusion of the
header, the< tuple > three templates aretemplate is available when any of the headers,< array > , or< ranges > are included.< utility >
3.17. Variants [variant]
Modify as follows.
synopsis [variant.syn]< variant > namespace std { // [ variant.variant ], class template variant template < class ... Types > class variant ; // [ variant.helper ], variant helper classes template < class T > struct variant_size ; // not defined template < class T > struct variant_size < const T > ; template < class T > struct variant_size < volatile T > ; template < class T > struct variant_size < const volatile T > ; template < class T > inline constexpr size_t variant_size_v = variant_size < T >:: value ; template < class ... Types > struct variant_size < variant < Types ... >> ; template < size_t I , class T > struct variant_alternative ; // not defined template < size_t I , class T > struct variant_alternative < I , const T > ; template < size_t I , class T > struct variant_alternative < I , volatile T > ; template < size_t I , class T > struct variant_alternative < I , const volatile T > ; [...] }
helper classes [variant.helper]variant template < class T > struct variant_size ; Remark: All specializations of
shall satisfy thevariant_size requirements with a base characteristic ofUnaryTypeTrait for someintegral_constant < size_t , N > .N template < class T > class variant_size < const T > ; template < class T > class variant_size < volatile T > ; template < class T > class variant_size < const volatile T > ; Let
denoteVS of the cv-unqualified typevariant_size < T > . ThenT each of the three templatesthe template shall satisfy therequirements with a base characteristic ofUnaryTypeTrait .integral_constant < size_t , VS :: value > template < class ... Types > struct variant_size < variant < Types ... >> : integral_constant < size_t , sizeof ...( Types ) > { }; template < size_t I , class T > class variant_alternative < I , const T > ; template < size_t I , class T > class variant_alternative < I , volatile T > ; template < size_t I , class T > class variant_alternative < I , const volatile T > ; Let
denoteVA of the cv-unqualified typevariant_alternative < I , T > . ThenT each of the three templatesthe template shall meet therequirements with a member typedefTransformationTrait that names the following type:type .add_const_t < VA :: type >
- for the first specialization,
,add_const_t < VA :: type > - for the second specialization,
, andadd_volatile_t < VA :: type > - for the third specialization,
.add_cv_t < VA :: type >
3.18. Atomic operations library [atomics]
Modify as follows.
Operations on atomic types [atomics.types.operations]
[ Note: Many operations are volatile-qualified. The "volatile as device register" semantics have not changed in the standard. This qualification means that volatility is preserved when applying these operations to volatile objects. It does not mean that operations on non-volatile objects become volatile. —end note ]
[...]
bool is_lock_free () const volatile noexcept ; bool is_lock_free () const noexcept ; Returns:
trueif the object’s operations are lock-free,falseotherwise.[ Note: The return value of the
member function is consistent with the value ofis_lock_free for the same type. —end note ]is_always_lock_free void store ( T desired , memory_order order = memory_order :: seq_cst ) volatile noexcept ; void store ( T desired , memory_order order = memory_order :: seq_cst ) noexcept ; Requires: The
argument shall not beorder ,memory_order :: consume , normemory_order :: acquire .memory_order :: acq_rel Effects: Atomically replaces the value pointed to by
Mandates: for thewith the value ofthis . Memory is affected according to the value ofdesired .order overload of this method,volatile isatomic < T >:: is_always_lock_free true.T operator = ( T desired ) volatile noexcept ; T operator = ( T desired ) noexcept ; Effects: Equivalent to
.store ( desired ) Returns:
.desired T load ( memory_order order = memory_order :: seq_cst ) const volatile noexcept ; T load ( memory_order order = memory_order :: seq_cst ) const noexcept ; Requires: The
argument shall not beorder normemory_order :: release .memory_order :: acq_rel Effects: Memory is affected according to the value of
.order Returns: Atomically returns the value pointed to by
Mandates: for the.this overload of this method,volatile isatomic < T >:: is_always_lock_free true.operator T () const volatile noexcept ; operator T () const noexcept ; Effects: Equivalent to:
return load (); T exchange ( T desired , memory_order order = memory_order :: seq_cst ) volatile noexcept ; T exchange ( T desired , memory_order order = memory_order :: seq_cst ) noexcept ; Effects: Atomically replaces the value pointed to by
withthis . Memory is affected according to the value ofdesired . These operations are atomic read-modify-write operations.order Returns: Atomically returns the value pointed to by
Mandates: for theimmediately before the effects.this overload of this method,volatile isatomic < T >:: is_always_lock_free true.bool compare_exchange_weak ( T & expected , T desired , memory_order success , memory_order failure ) volatile noexcept ; bool compare_exchange_weak ( T & expected , T desired , memory_order success , memory_order failure ) noexcept ; bool compare_exchange_strong ( T & expected , T desired , memory_order success , memory_order failure ) volatile noexcept ; bool compare_exchange_strong ( T & expected , T desired , memory_order success , memory_order failure ) noexcept ; bool compare_exchange_weak ( T & expected , T desired , memory_order order = memory_order :: seq_cst ) volatile noexcept ; bool compare_exchange_weak ( T & expected , T desired , memory_order order = memory_order :: seq_cst ) noexcept ; bool compare_exchange_strong ( T & expected , T desired , memory_order order = memory_order :: seq_cst ) volatile noexcept ; bool compare_exchange_strong ( T & expected , T desired , memory_order order = memory_order :: seq_cst ) noexcept ; Requires: The
argument shall not befailure normemory_order :: release .memory_order :: acq_rel Effects: Retrieves the value in
. It then atomically compares the value representation of the value pointed to byexpected for equality with that previously retrieved fromthis ,eand if true, replaces the value pointed to byexpected with that inthis . If and only if the comparison is true, memory is affected according to the value ofdesired , and if the comparison is false, memory is affected according to the value ofsuccess . When only onefailure argument is supplied, the value ofmemory_order issuccess , and the value oforder isfailure except that a value oforder shall be replaced by the valuememory_order :: acq_rel and a value ofmemory_order :: acquire shall be replaced by the valuememory_order :: release . If and only if the comparison is false then, after the atomic operation, the value inmemory_order :: relaxed is replaced by the value pointed to byexpected during the atomic comparison. If the operation returnsthis true, these operations are atomic read-modify-write operations on the memory pointed to by. Otherwise, these operations are atomic load operations on that memory.this Returns: The result of the comparison.
Mandates: for theoverloads of these methods,volatile isatomic < T >:: is_always_lock_free true.[...]
Specializations for integers [atomics.types.int]
T fetch_ key ( T operand , memory_order order = memory_order :: seq_cst ) volatile noexcept ; T fetch_ key ( T operand , memory_order order = memory_order :: seq_cst ) noexcept ; Effects: Atomically replaces the value pointed to by
with the result of the computation applied to the value pointed to bythis and the giventhis . Memory is affected according to the value ofoperand . These operations are atomic read-modify-write operations.order Returns: Atomically, the value pointed to by
Mandates: for theimmediately before the effects.this overloads of these methods,volatile isatomic < T >:: is_always_lock_free true.Remarks: For signed integer types, the result is as if the object value and parameters were converted to their corresponding unsigned types, the computation performed on those types, and the result converted back to the signed type. [ Note: There are no undefined results arising from the computation. —end note ]
T operator op = ( T operand ) volatile noexcept ; T operator op = ( T operand ) noexcept ; Effects: Equivalent to:
Mandates: for thereturn fetch_ key ( operand ) op operand ; overloads of these methods,volatile isatomic < T >:: is_always_lock_free true.Specializations for floating-point types [atomics.types.float]
The following operations perform arithmetic addition and subtraction computations. The key, operator, and computation correspondence are identified in [atomic.arithmetic.computations].
T A :: fetch_ key ( T operand , memory_order order = memory_order_seq_cst ) volatile noexcept ; T A :: fetch_ key ( T operand , memory_order order = memory_order_seq_cst ) noexcept ; Effects: Atomically replaces the value pointed to by
with the result of the computation applied to the value pointed to bythis and the giventhis . Memory is affected according to the value ofoperand . These operations are atomic read-modify-write operations.order Returns: Atomically, the value pointed to by
Mandates: for theimmediately before the effects.this overloads of these methods,volatile isatomic < T >:: is_always_lock_free true.Remarks: If the result is not a representable value for its type the result is unspecified, but the operations otherwise have no undefined behavior. Atomic arithmetic operations on
should conform to thefloating - point traits associated with the floating-point type. The floating-point environment for atomic arithmetic operations onstd :: numeric_limits < floating - point > may be different than the calling thread’s floating-point environment.floating - point T operator op = ( T operand ) volatile noexcept ; T operator op = ( T operand ) noexcept ; Effects: Equivalent to:
return fetch_ key ( operand ) op operand ; Remarks: If the result is not a representable value for its type the result is unspecified, but the operations otherwise have no undefined behavior. Atomic arithmetic operations on
should conform to thefloating - point traits associated with the floating-point type. The floating-point environment for atomic arithmetic operations onstd :: numeric_limits < floating - point > may be different than the calling thread’s floating-point environment.floating - point Partial specialization for pointers [atomics.types.pointer]
T * fetch_ key ( ptrdiff_t operand , memory_order order = memory_order :: seq_cst ) volatile noexcept ; T * fetch_ key ( ptrdiff_t operand , memory_order order = memory_order :: seq_cst ) noexcept ; Requires: T shall be an object type, otherwise the program is ill-formed. [ Note: Pointer arithmetic on
or function pointers is ill-formed. —end note ]void * Effects: Atomically replaces the value pointed to by
with the result of the computation applied to the value pointed to bythis and the giventhis . Memory is affected according to the value ofoperand . These operations are atomic read-modify-write operations.order Returns: Atomically, the value pointed to by
Mandates: for theimmediately before the effects.this overloads of these methods,volatile isatomic < T >:: is_always_lock_free true.Remarks: The result may be an undefined address, but the operations otherwise have no undefined behavior.
T * operator op = ( ptrdiff_t operand ) volatile noexcept ; T * operator op = ( ptrdiff_t operand ) noexcept ; Effects: Equivalent to:
return fetch_ key ( operand ) op operand ; Member operators common to integers and pointers to objects [atomics.types.memop]
T operator ++ ( int ) volatile noexcept ; T operator ++ ( int ) noexcept ; Effects: Equivalent to:
return fetch_add ( 1 ); T operator -- ( int ) volatile noexcept ; T operator -- ( int ) noexcept ; Effects: Equivalent to:
return fetch_sub ( 1 ); T operator ++ () volatile noexcept ; T operator ++ () noexcept ; Effects: Equivalent to:
return fetch_add ( 1 ) + 1 ; T operator -- () volatile noexcept ; T operator -- () noexcept ; Effects: Equivalent to:
return fetch_sub ( 1 ) - 1 ; Non-member functions [atomics.nonmembers]
A non-member function template whose name matches the pattern
or the patternatomic_ f invokes the member functionatomic_ f _explicit , with the value of the first parameter as the object expression and the values of the remaining parameters (if any) as the arguments of the member function call, in order. An argument for a parameter of typef is dereferenced when passed to the member function call. If no such member function exists, the program is ill-formed.atomic < T >:: value_type * template < class T > void atomic_init ( volatile atomic < T >* object , typename atomic < T >:: value_type desired ) noexcept ; template < class T > void atomic_init ( atomic < T >* object , typename atomic < T >:: value_type desired ) noexcept ; Effects: Non-atomically initializes
Mandates: for thewith value* object . This function shall only be applied to objects that have been default constructed, and then only once. [ Note: These semantics ensure compatibility with C. —end note ] [ Note: Concurrent access from another thread, even via an atomic operation, constitutes a data race. —end note ]desired overload of this method,volatile isatomic < T >:: is_always_lock_free true.[ Note: The non-member functions enable programmers to write code that can be compiled as either C or C++, for example in a shared header file. —end note ]
3.19. Annex D
All mentions of deprecation in language clauses, and all deletions in library
clauses above should be added to Annex D under [depr.volatile], such that is now deprecated in these use cases.