Refactor atomic reduction sequence replacements into tables.
Provide alternative wording for atomic reduction sequence replacement using "as if" integer operations to provide same valid replacements as for integer reduction operations.
R2: post Wroclaw: simplifies reduction sequence wording.
R1: post St. Louis '24
Overview of changes: incorporate SG6 and SG1 feedback into the proposal, which resolved all open questions. Restructure exposition about what's being proposed accordingly.
[SG6 Review]:
Forward it; does not need to look at it again.
Agree on "generalized" reductions being the default and not providing version without.
Do not use GENERALIZED_SUM to specify non-associative floating-point atomic reduction operations since it does not make sense for two values. Instead, attempt to add a std::non_associative_add operation that is exempted from the C standard clause in Annex F that makes re-ordering non-conforming. Recommended discussing this with CWG, which I did, and caused us to end up pursuing a different alternative (defining "reduction sequences") which is pursued in R1 instead.
[SG1 Review]: Add wording for reduction sequences.
[SG1 Review]: Add compare_store operation.
[SG1 Review]: Update to handle fenv and errno for floating-point atomics.
[SG1 Review]: Make "generalized" atomic floating-point reductions the default and do not provide fallback (beyond fetch_<op>).
[SG1 Review]: Provide unsequenced support.
[SG1 Review]: polls taken:
Reduction operations are not a step, but infinite loops around reductions operations are not UB (practically implementations can assume reductions are finite)
SF
F
N
A
SA
3
9
1
0
0
The single floating point reduction operations should allow "fast math" (allow tree reductions), you can get precise results using fetch_add/sub
SF
F
N
A
SA
5
5
1
1
0
Any objection to approve the design of P3111R0: No objection.
R0: initial revision.
Atomic Reduction Operations
Atomic Reduction Operations are atomic read-modify-write (RMW) operations (like fetch_add) that do not "fetch" the old value and are not reads for the purpose of building synchronization with acquire fences. This enables implementations to leverage hardware acceleration available in modern CPU and GPU architectures.
Furthermore, we propose to allow atomic memory operations that aren't reads in unsequenced execution, and to extend atomic arithmetic reduction operations for floating-point types with operations that assume floating-point arithmetic is associative.
Introduction
Concurrent algorithms performing atomic RMW operations that discard the old fetched value are very common in high-performance computing, e.g., finite-element matrix assembly, data analytics (e.g. building histograms), etc.
Consider the following parallel algorithm to build a histogram (full implementation):
Correctness (undefined behavior): The program should have used the par execution policy to avoid undefined behavior since the atomic operation is not:
Potentially concurrent and therefore exhibits a data-race in unsequenced contexts ([intro.execution]).
Vectorization-safe [algorithms.parallel.defns#5] since it is specificed to synchronize with other function invocations.
Performance: Sophisticated compiler analysis required to optimize the above program for scalable hardware architectures with atomic reduction operations.
Atomic reduction operations address both shortcomings:
Some of these instructions lack a destination operand (red, STADD, AADD). Others change semantics if the destination register used discards the result (Arm's LDADD RZ, SWP RZ, CAS RZ).
All ISAs provide the same sematics: these are not loads from the point-of-view of the Memory Model, and therefore do not participate in acquire sequences, but they do participate in release sequences:
These architectures provide both "relaxed" and "release" orderings for the reductions (e.g. red.relaxed/red.release, STADD/STADDL).
Performance
On hardware architectures that implement these as far atomics, the exposed latency of Atomic Reduction Operations may be as low as half that of "fetch_<key>" operations.
Example: on an NVIDIA Hopper H100 GPU, replacing atomic.fetch_add with atomic.reduce_add on the Histogram Example (Example 0) improves throughput by 1.2x.
Furthermore, non-associative floating-point atomic operations, like fetch_add, are required to read the "latest value", which sequentializes their execution. In the following example, the outcome x == a + (b + c) is not allowed, because either the atomic operation of thread0 happens-before that of thread1, or vice-versa, and floating-point arithmetic is not associative:
// Litmus test 2:
atomic<float> x = a;voidthread0(){ x.fetch_add(b, memory_order_relaxed);}voidthread1(){ x.fetch_add(c, memory_order_relaxed);}
Allowing the x == a + (b + c) outcome enables implementations to perform a tree-reduction, which improves complexity from O(N) to O(log(N)), at negligible higher amount of non-determinism which is already inherent to the use of atomic operations:
// Litmus test 3:
atomic<float> x = a;
x.reduce_add(b, memory_order_relaxed);
x.reduce_add(c, memory_order_relaxed);// Sound to merge these two operations into one:// x.reduce_add(b + c);
On GPU architectures, performing an horizontal reduction for then issuing a single atomic operation per thread group, reduces the number of atomic operation issued by up to the size of the thread group.
Functionality
Currently, all atomic memory operations are vectorization-unsafe and therefore not allowed in element access functions of parallel algorithms when the unseq or par_unseq execution policies are used (see [algorithms.parallel.exec.5] and [algorithms.parallel.exec.7]). Atomic memory operations that "read" (e.g. load, fetch_<key>, compare_exchange, exchange, …) enable building synchronization edges that block, which within unseq/par_unseq leads to dead-locks. N4070 solved this by tightening the wording to disallow any synchronization API from being called from within unseq/par_unseq.
Allowing Atomic Writes and Atomic Reduction Operations in unsequenced execution increases the set of concurrent algorithms that can be implemented in the lowest-common denominator of hardware that C++ supports. In particular, many hardware architectures that can accelerate unseq/par_unseq but cannot accelerate par (e.g. most non-NVIDIA GPUs), provide acceleration for atomic reduction operations.
We propose to make lock-free atomic operations that are not reads vectorization safe to enable calling them from unsequenced execution. Atomic operations that read remain vectorization-unsafe and therefore UB:
Design
For each atomic fetch_{OP} in the atomic<T> and atomic_ref<T> class templates and their specializations, we introduce new reduce_{OP} member functions that return void:
template<classT>structatomic_ref{
T fetch_add(T v, memory_order o)constnoexcept;voidreduce_add(T v, memory_order o)constnoexcept;};
We also introduce compare_store, which is the Atomic Reduction Operation analogous of compare_exchange:
template<classT>structatomic_ref{boolcompare_exchange_weak(T& expected, T desired, memory_order o)constnoexcept;voidcompare_store(T expected, T desired, memory_order o)constnoexcept;};
The reduce_{OP} and compare_store APIs are vectorization safe if the atomic is_always_lock_free is true, i.e., they are allowed in Element Access Functions ([algorithms.paralle.defns]) of parallel algorithms:
Furthermore, we specify non-associative floating-point atomic reduction operations to enable tree-reduction implementations that improve complexity from O(N) to O(log(N)) by minimally increasing the non-determinism which is already inherent to atomic operations. This allows producing the x == a + (b + c) outcome in the following example, which enables the optimization that merges the two reduce_add operations into a single one:
atomic<float> x = a;
x.reduce_add(b, memory_order_relaxed);
x.reduce_add(c, memory_order_relaxed);// Sound to merge these two operations into one:// x.store_add(b + c);
Applications that need the sequential semantics can use fetch_add instead.
Implementation impact
It is correct to conservatively implement reduce_{OP} as a call to fetch_{OP}. We evaluated the following implementations of unsequenced execution policies, which are not impacted:
OpenMP simd pragma for unseq and par_unseq, since OpenMP supports atomics within simd regions.
Enable Atomic Reductions as fetch_<key> optimizations
Attempting to improve application performance by implementing compiler-optimizations to leverage Atomic Reduction Operations from fetch_<key> APIs whose result is unused has become a rite of passage for compiler engineers, e.g., GCC#509632, LLVM#68428, LLVM#72747, … Unfortunately, "simple" optimization strategies break backward compatibility in the following litmus tests (among others).
Litmus Test 0: from Will Deacon. Performing the optimization to replace the introduces the y == 2 && r0 == 1 && r1 == 0 outcome:
In some architectures, Atomic Reduction Operations can write to memory pages or memory locations that are not readable, e.g., MMIO registers on NVIDIA GPUs, and need a reliable programming model that does not depend on compiler-optimizations for functionality.
Naming
We considered the following alternative names:
atomic.add: breaks for logical operations (e.g. atomic.and, etc.).
atomic.store_add: clearly states that this operation is a "store" (and not a "load").
atomic.reduce_add: clearly state what this operation is for (reductions).
We considered providing separate version of non-associative floating-point atomic reduction operations with and without the support for tree-reductions, e.g., atomic.reduce_add (no tree-reduction support) and atomic.reduce_add_generalized (tree-reduction support), but decided against it because atomic operations are inherently non-deterministic, this relaxation only minimally impacts that, and fetch_add already provides (and has to continue to provide) the sequential semantics without this relaxation.
Memory Ordering
We choose to support memory_order_relaxed, memory_order_release, and memory_order_seq_cst.
We may need a note stating that replacing the operations in a reduction sequence is only valid as long as the replacement maintains the other ordering properties of the operations as defined in [intro.races]. Examples:
Unresolved question: Are tree reductions only supported for memory_order_relaxed? No, see litmus tests for release and seq_cst.
Formalization
Herd already support these for STADD on Arm, and the NVIDIA Volta Memory Model supports these for red and multimem.red on PTX. If we decide to pursue this exposure direction, this proposal would benefit from extending Herd's RC11 with reduction sequences for floating-point.
Wording
Do NOT modify [intro.execution.10]!
We do not need to modify [intro.execution.10] to enable using atomic reduction operations in unsequenced contexts, because this section does not prevent that: atomic.foo() are function calls and two function calls are always indeterminately sequenced, not unsequenced. That is, function calls never overlap, and this section does not impact that.
Except where noted, evaluations of operands of individual operators and of subexpressions of individual expressions are unsequenced. [Note 5: In an expression that is evaluated more than once during the execution of a program, unsequenced and indeterminately sequenced evaluations of its subexpressions need not be performed consistently in different evaluations. — end note]
The value computations of the operands of an operator are sequenced before the value computation of the result of the operator. If a side effect on a memory location ([intro.memory]) is unsequenced relative to either another side effect on the same memory location or a value computation using the value of any object in the same memory location, and they are not lock-free atomic read operations ([atomics]) or potentially concurrent ([intro.multithread]), the behavior is undefined. [Note 6: The next subclause imposes similar, but more complex restrictions on potentially concurrent computations. — end note]
[Example 3:
voidg(int i){
i =7, i++, i++;// i becomes 9
i = i+++1;// the value of i is incremented
i = i+++ i;// undefined behavior
i = i +1;// the value of i is incremented}
Review Note: the intent of this wording change is to enable tree-reduction implementations for non-associative floating-point atomic reduction operations which, in this proposal, are only reduce_add and reduce_sub. SG1 agreed that the single floating point reduction operations should allow "fast math" (allow tree reductions) because programs can get precise results using fetch_add/sub. Alternative wording strategies using GENERALIZED_NONCOMMUTATIVE_SUM were explored but are not viable since only two values are involved. Review Note: Since other proposals in flight add fetch_mul and fetch_div, and those would need to extend this, this revision of this paper shows how that would look like by introducing "multiplicative reduction sequences". Review Note: For integers, implementations are already allowed to perform tree reductions, as long as they don't violate any other ordering rules. The only goal of the following reduction sequence wording is to allow the exact same thing for floats, since implementation cannot perform tree reductions because the ops are not associative and impact the results. Maybe instead of adding all of this wording, we could instead say that reduction operations on floats may be re-ordered or re-associated "as if" they were operations on integers? For example, we could add a Remark to the floating-point specialization of atomic and atomic_ref that just says: "Remark: The associativity of floating-point atomic reduction operations is the same as that of integers."
Option A.0: this option specifies the optimization that is allowed for floating-point atomic reduction operations..
A reduction sequence is a maximal contiguous sub-sequence of side effects in the modification order of M, where each operation is an atomic reduction operation. If replacing two adjacent integer operations in a reduction sequence is not observable, this same replacement may be performed for two adjacent floating-point operations in a reduction sequence. [Note 1: Combining adjacent operations in a reduction sequence is often not observable for integer operations but is often observable for floating-point operations due to associativity. This paragraph enables implementations to optimize floating-point atomic reduction operations using tree reductions. - end note]
Alternative A.1: Replace the last sentence in the paragraph with: "Any two adjacent floating-point operations in a reduction sequence can be replaced by a single operation as if the operations were integer operations.".
Option B: this option explicitly specify the allowed replacements for floating-point operations using a table.
A reduction sequence is a maximal contiguous sub-sequence of side effects in the modification order of M, where each operation is an atomic reduction operation. Any two adjacent floating-point operations in a reduction sequence whose replacement does not contradict any memory ordering requirements can be replaced by a single operation, recursively, as specified in Table [tab.red.seq.replacement]. The replaced operation uses the largest memory order of both operations according to memory_order_relaxed < memory_order_release < memory_order_seq_cst. For signed integer types, the intermediate computation is performed as if the objects a and b were converted to their corresponding unsigned types, and its result converted back to the signed type. [Note 1: Replacing any two adjacent operations in a reduction sequence is only valid if the replacement maintains the other ordering properties of the operations as defined in [intro.races]. The assertion in the following example holds:
is not allowed, but replacing them before the store as follows:
end note]
Table [tab.red.seq.replacement]: Allowed replacements of two adjacent floating-point atomic reduction operations reduce_<key> within a floating-point atomic reduction sequence on memory location M, where the value parameter of the first atomic operation is a, and that of the second is b.
Second Op First Op
add
sub
min
max
add
add(a + b)
add(a - b)
n/a
n/a
sub
add(b - a)
sub(a + b)
n/a
n/a
min
n/a
n/a
min(min(a,b))
n/a
max
n/a
n/a
n/a
max(max(a,b))
Option B.1: this option may be extended to mul and div operations as follows:
Second Op First Op
mul
div
mul
mul(a * b)
mul(a / b)
div
mul(b / a)
div(a * b)
Option B.2: this option explicitly specify the allowed replacements for floating-point operations without a table.
… as specified below, recursively:
M.reduce_add(a, o0) and M.reduce_add(b, o1) can be replaced with M.reduce_add(a + b, max_order(o0, o1)),
M.reduce_sub(a, o0) and M.reduce_sub(b, o1) can be replaced with M.reduce_sub(a + b, max_order(o0, o1)),
M.reduce_add(a, o0) and M.reduce_sub(b, o1) can be replaced with M.reduce_add(a - b, max_order(o0, o1)),
M.reduce_sub(a, o0) and M.reduce_add(b, o1) can be replaced with M.reduce_add(b - a, max_order(o0, o1)),
M.reduce_mul(a, o0) and M.reduce_mul(b, o1) can be replaced with M.reduce_mul(a * b, max_order(o0, o1)),
M.reduce_div(a, o0) and M.reduce_div(b, o1) can be replaced with M.reduce_div(a * b, max_order(o0, o1)),
M.reduce_mul(a, o0) and M.reduce_div(b, o1) can be replaced with M.reduce_mul(a / b, max_order(o0, o1)),
M.reduce_div(a, o0) and M.reduce_mul(b, o1) can be replaced with M.reduce_mul(b / a, max_order(o0, o1)),
M.reduce_min(a, o0) and M.reduce_min(b, o1) can be replaced with M.reduce_min(min(a, b), max_order(o0, o1)),
M.reduce_max(a, o0) and M.reduce_max(b, o1) can be replaced with M.reduce_max(max(a, b), max_order(o0, o1)).
Review note: the first bullet says which standard library functions are, in general, vectorization unsafe, and the second bullet carves out exceptions. Review note: the current wording intent is for fetch_add(relaxed) to be vectorization-unsafe, but the standard words in [intro.execution.10] (see above) that were supposed to fix that, do not. So we currently ban those here as a drive-by change that should be filled as LWG defect report, but that this proposal would then need to carve out an exception for atomic reduction operations. Review Note: SG1 requested making "relaxed", "release", and "seq_cst" atomic reduction operations not be vectorization unsafe, and that is the intent of the words below.
A standard library function is vectorization-unsafe if:
it is specified to synchronize with another function invocation, or another function invocation is specified to synchronize with it,andor it is a relaxed atomic operation [atomics.order], and
if it is not a memory allocation or deallocation function or lock-free atomic reduction operation.
[Note 2: Implementations must ensure that internal synchronization inside standard library functions does not prevent forward progress when those functions are executed by threads of execution with weakly parallel forward progress guarantees. — end note]
[Example 2:
int x =0;
std::mutex m;voidf(){int a[]={1,2};
std::for_each(std::execution::par_unseq, std::begin(a), std::end(a),[&](int){
std::lock_guard<mutex>guard(m);// incorrect: lock_guard constructor calls m.lock()++x;});}
The above program may result in two consecutive calls to m.lock() on the same thread of execution (which may deadlock), because the applications of the function object are not guaranteed to run on different threads of execution. — end example]
Review Note: SG1 design intent is that Reduction operations are not a step, but infinite loops around reductions operations are not UB (practically implementations can assume reductions are finite). Gonzalo's note: such loops cause all threads to loose forward progress. Review Note: this change makes Atomic Reduction Operations and Atomic Stores/Fences asymmetric, but making this change for Atomic Stores/Fences is a breaking change to be pursued in a separate paper per SG1 feedback. When adding these new operations, it does make sense to make this change, to avoid introducing undesired semantics that would be a breaking change to fix.
The implementation may assume that any thread will eventually do one of the following:
terminate,
invoke the function std::this_thread::yield ([thread.thread.this]),
make a call to a library I/O function,
perform an access through a volatile glvalue, or
perform a synchronization operation or an atomic operation other than an atomic reduction operation [atomics.order], or
continue execution of a trivial infinite loop ([stmt.iter.general]).
[Note 1: This is intended to allow compiler transformations such as removal of empty loops, even when termination cannot be proven. — end note]
Review note: same note as above about exempting atomic stores in a separate paper.
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 volatile glvalue, or
completion of a call to a library I/O function, or a synchronization operation, or an atomic operation other than an atomic reduction operation [atomics.order].
This subclause introduces synchronization primitives called fences. Fences can have acquire semantics, release semantics, or both. A fence with acquire semantics is called an acquire fence. A fence with release semantics is called a release fence.
A release fence A synchronizes with an acquire fence B if there exist atomic operations X and Y, where Y is not an atomic reduction operation [atomics.order], both operating on some atomic object M, such that A is sequenced before X, X modifies M, Y is sequenced before B, and Y reads the value written by X or a value written by any side effect in the hypothetical release sequence X would head if it were a release operation.
A release fence A synchronizes with an atomic operation B that performs an acquire operation on an atomic object M if there exists an atomic operation X such that A is sequenced before X, X modifies M, and B reads the value written by X or a value written by any side effect in the hypothetical release sequence X would head if it were a release operation.
An atomic operation A that is a release operation on an atomic object M synchronizes with an acquire fence B if there exists some atomic operation X on M such that X is sequenced before B and reads the value written by A or a value written by any side effect in the release sequence headed by A.
Atomic Reduction Operation APIs
The following operations perform arithmetic computations. The correspondence among key, operator, and computation is specified in Table 145:
void compare_store(T expected, T desired,
memory_order order = memory_order::seq_cst) const noexcept;
Preconditions: order is memory_order::relaxed, memory_order::release, or memory_order::seq_cst.
Effects: Atomically compares the value representation of the value referenced by *ptr for equality with the value in expected, and if true, replaces the value pointed to by *ptr with that in desired. If and only if the comparison is true, memory is affected according to the value of order; otherwise, memory is not affected. This operation is an atomic read-modify-write operations ([intro.races]) and an atomic reduction operations on the value referenced by *ptr.
void reduce_<key>(integral-type operand,
memory_order order = memory_order_seq_cst) const noexcept;
Preconditions: order is memory_order_relaxed, memory_order_release, or memory_order_seq_cst.
Effects: Atomically replaces the value referenced by *ptr with the result of the computation applied to the value referenced by *ptr and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.
Remarks: Except for reduce_max and reduce_min, 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 2: There are no undefined results arising from the computation. — end note] [Note 3: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note] For reduce_max and reduce_min, the maximum and minimum computation is performed as if by max and min algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.
Review note: add reduce_minimum, _minimum_num, _maximum, and _maximum_num once 3008 is merged.
void reduce_key(floating-point-type operand,
memory_order order = memory_order::seq_cst) const noexcept;
Preconditions: order is memory_order_relaxed, memory_order_release, or memory_order_seq_cst.
Effects: Atomically replaces the value referenced by *ptr with the result of the computation applied to the value referenced by *ptr and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.
Remarks: If the result is not a representable value for its type ([expr.pre]), the result is unspecified, but the operations otherwise have no undefined behavior. Atomic arithmetic operations on floating-point-type should conform to the std::numeric_limits<floating-point-type> traits associated with the floating-point type ([limits.syn]). The floating-point environment ([cfenv]) for atomic arithmetic operations on floating-point-type may be different than the calling thread's floating-point environment.
[Note 3: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note]
void reduce_key(difference_type operand,
memory_order order = memory_order::seq_cst) const noexcept;
Mandates: T is a complete object type.ins>
Preconditions: order is memory_order_relaxed, memory_order_release, or memory_order_seq_cst.
Effects: Atomically replaces the value referenced by *ptr with the result of the computation applied to the value referenced by *ptr and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.
Remarks: The result may be an undefined address, but the operations otherwise have no undefined behavior. For fetch_max and fetch_min, the maximum and minimum computation is performed as if by max and min algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments. [Note 1: If the pointers point to different complete objects (or subobjects thereof), the < operator does not establish a strict weak ordering (Table 29, [expr.rel]). — end note] [Note 2: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note]
void compare_store(T expected, T desired,
memory_order order = memory_order::seq_cst) const noexcept;
Preconditions: order is memory_order::relaxed, memory_order::release, or memory_order::seq_cst.
Effects: Atomically compares the value representation of the value pointed to by this for equality with that of expected, and if true, replaces the value pointed to by this with that in desired. If and only if the comparison is true, memory is affected according to the value of order; otherwise, memory is not affected. This operation is an atomic read-modify-write operations ([intro.races]) and an atomic reduction operations on the memory pointed to by this.
void reduce_<key>(integral-type operand,
memory_order order = memory_order_seq_cst) volatile noexcept;
void reduce_<key>(integral-type operand,
memory_order order = memory_order_seq_cst) noexcept;
Constraints: For the volatile overload of this function, is_always_lock_free is true.
Preconditions: order is memory_order_relaxed, memory_order_release, or memory_order_seq_cst.
Effects: Atomically replaces the value pointed to by this with the result of the computation applied to the value pointed to by this and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.multithread]) and atomic reduction operations.
Remarks: Except for reduce_max and reduce_min, 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 2: There are no undefined results arising from the computation. — end note]
[Note 3: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note] For reduce_max and reduce_min, the maximum and minimum computation is performed as if by max and min algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.
Review note: add reduce_minimum, _minimum_num, _maximum, and _maximum_num once 3008 is merged.
void reduce_key(T operand,
memory_order order = memory_order::seq_cst) volatile noexcept;
void reduce_key(T operand,
memory_order order = memory_order::seq_cst) noexcept;
Constraints: For the volatile overload of this function, is_always_lock_free is true.
Preconditions: order is memory_order_relaxed, memory_order_release, or memory_order_seq_cst.
Effects: Atomically replaces the value pointed to by this with the result of the computation applied to the value pointed to by this and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.multithread]) and atomic reduction operations.
Remarks: If the result is not a representable value for its type ([expr.pre]) the result is unspecified, but the operations otherwise have no undefined behavior. Atomic arithmetic operations on floating-point-type should conform to the std::numeric_limits<floating-point-type> traits associated with the floating-point type ([limits.syn]). The floating-point environment ([cfenv]) for atomic arithmetic operations on floating-point-type may be different than the calling thread's floating-point environment.
void reduce_key(ptrdiff_t operand,
memory_order order = memory_order::seq_cst) volatile noexcept;
void reduce_key(ptrdiff_t operand,
memory_order order = memory_order::seq_cst) noexcept;
Constraints: For the volatile overload of this function, is_always_lock_free is true.
Mandates: T is a complete object type. [Note 1: Pointer arithmetic on void* or function pointers is ill-formed. — end note]
Effects: Atomically replaces the value pointed to by this with the result of the computation applied to the value pointed to by this and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.multithread]).
Remarks: The result may be an undefined address, but the operations otherwise have no undefined behavior. For reduce_max and reduce_min, the maximum and minimum computation is performed as if by max and min algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments. [Note 2: If the pointers point to different complete objects (or subobjects thereof), the < operator does not establish a strict weak ordering (Table 29, [expr.rel]). — end note]
Document number: P3111R3.
Date: 2025-01-13.
Authors: Gonzalo Brito Gadeschi, Simon Cooksey, Daniel Lustig.
Reply to: Gonzalo Brito Gadeschi <gonzalob _at_ nvidia.com>.
Audience: EWG, LEWG.
Table of Contents
Changelog
GENERALIZED_SUM
to specify non-associative floating-point atomic reduction operations since it does not make sense for two values. Instead, attempt to add astd::non_associative_add
operation that is exempted from the C standard clause in Annex F that makes re-ordering non-conforming. Recommended discussing this with CWG, which I did, and caused us to end up pursuing a different alternative (defining "reduction sequences") which is pursued in R1 instead.compare_store
operation.fenv
anderrno
for floating-point atomics.fetch_<op>
).Reduction operations are not a step, but infinite loops around reductions operations are not UB (practically implementations can assume reductions are finite)
The single floating point reduction operations should allow "fast math" (allow tree reductions), you can get precise results using fetch_add/sub
Any objection to approve the design of P3111R0: No objection.
Atomic Reduction Operations
Atomic Reduction Operations are atomic read-modify-write (RMW) operations (like
fetch_add
) that do not "fetch" the old value and are not reads for the purpose of building synchronization with acquire fences. This enables implementations to leverage hardware acceleration available in modern CPU and GPU architectures.Furthermore, we propose to allow atomic memory operations that aren't reads in unsequenced execution, and to extend atomic arithmetic reduction operations for floating-point types with operations that assume floating-point arithmetic is associative.
Introduction
Concurrent algorithms performing atomic RMW operations that discard the old fetched value are very common in high-performance computing, e.g., finite-element matrix assembly, data analytics (e.g. building histograms), etc.
Consider the following parallel algorithm to build a histogram (full implementation):
This program has two main issues:
par
execution policy to avoid undefined behavior since the atomic operation is not:Atomic reduction operations address both shortcomings:
This new operation can then be used in the Histogram Example (example 0), to replace the
fetch_add
withreduce_add
.Motivation
Hardware Exposure
The following ISAs provide Atomic Reduction Operation:
red
.LDADD RZ
,STADD
,SWP RZ
,CAS RZ
.Some of these instructions lack a destination operand (
red
,STADD
, AADD). Others change semantics if the destination register used discards the result (Arm'sLDADD RZ
,SWP RZ
,CAS RZ
).All ISAs provide the same sematics: these are not loads from the point-of-view of the Memory Model, and therefore do not participate in acquire sequences, but they do participate in release sequences:
red
is "not a read operation".These architectures provide both "relaxed" and "release" orderings for the reductions (e.g.
red.relaxed
/red.release
,STADD
/STADDL
).Performance
On hardware architectures that implement these as far atomics, the exposed latency of Atomic Reduction Operations may be as low as half that of "
fetch_<key>
" operations.Example: on an NVIDIA Hopper H100 GPU, replacing
atomic.fetch_add
withatomic.reduce_add
on the Histogram Example (Example 0) improves throughput by 1.2x.Furthermore, non-associative floating-point atomic operations, like
fetch_add
, are required to read the "latest value", which sequentializes their execution. In the following example, the outcomex == a + (b + c)
is not allowed, because either the atomic operation of thread0 happens-before that of thread1, or vice-versa, and floating-point arithmetic is not associative:Allowing the
x == a + (b + c)
outcome enables implementations to perform a tree-reduction, which improves complexity fromO(N)
toO(log(N))
, at negligible higher amount of non-determinism which is already inherent to the use of atomic operations:On GPU architectures, performing an horizontal reduction for then issuing a single atomic operation per thread group, reduces the number of atomic operation issued by up to the size of the thread group.
Functionality
Currently, all atomic memory operations are vectorization-unsafe and therefore not allowed in element access functions of parallel algorithms when the
unseq
orpar_unseq
execution policies are used (see [algorithms.parallel.exec.5] and [algorithms.parallel.exec.7]). Atomic memory operations that "read" (e.g.load
,fetch_<key>
,compare_exchange
,exchange
, …) enable building synchronization edges that block, which withinunseq
/par_unseq
leads to dead-locks. N4070 solved this by tightening the wording to disallow any synchronization API from being called from withinunseq
/par_unseq
.Allowing Atomic Writes and Atomic Reduction Operations in unsequenced execution increases the set of concurrent algorithms that can be implemented in the lowest-common denominator of hardware that C++ supports. In particular, many hardware architectures that can accelerate
unseq
/par_unseq
but cannot acceleratepar
(e.g. most non-NVIDIA GPUs), provide acceleration for atomic reduction operations.We propose to make lock-free atomic operations that are not reads vectorization safe to enable calling them from unsequenced execution. Atomic operations that read remain vectorization-unsafe and therefore UB:
Design
For each atomic
fetch_{OP}
in theatomic<T>
andatomic_ref<T>
class templates and their specializations, we introduce newreduce_{OP}
member functions that returnvoid
:We also introduce
compare_store
, which is the Atomic Reduction Operation analogous ofcompare_exchange
:The
reduce_{OP}
andcompare_store
APIs are vectorization safe if the atomicis_always_lock_free
is true, i.e., they are allowed in Element Access Functions ([algorithms.paralle.defns]) of parallel algorithms:Furthermore, we specify non-associative floating-point atomic reduction operations to enable tree-reduction implementations that improve complexity from
O(N)
toO(log(N))
by minimally increasing the non-determinism which is already inherent to atomic operations. This allows producing thex == a + (b + c)
outcome in the following example, which enables the optimization that merges the tworeduce_add
operations into a single one:Applications that need the sequential semantics can use
fetch_add
instead.Implementation impact
It is correct to conservatively implement
reduce_{OP}
as a call tofetch_{OP}
. We evaluated the following implementations of unsequenced execution policies, which are not impacted:unseq
andpar_unseq
, since OpenMP supports atomics withinsimd
regions.Design Alternatives
Enable Atomic Reductions as
fetch_<key>
optimizationsAttempting to improve application performance by implementing compiler-optimizations to leverage Atomic Reduction Operations from
fetch_<key>
APIs whose result is unused has become a rite of passage for compiler engineers, e.g., GCC#509632, LLVM#68428, LLVM#72747, … Unfortunately, "simple" optimization strategies break backward compatibility in the following litmus tests (among others).Litmus Test 0: from Will Deacon. Performing the optimization to replace the introduces the
y == 2 && r0 == 1 && r1 == 0
outcome:Litmus Test 1: from Luke Geeson. Performing the optimization of replacing the exchange with a store introduces the
r0 == 0 && y == 2
outcome:In some architectures, Atomic Reduction Operations can write to memory pages or memory locations that are not readable, e.g., MMIO registers on NVIDIA GPUs, and need a reliable programming model that does not depend on compiler-optimizations for functionality.
Naming
We considered the following alternative names:
atomic.add
: breaks for logical operations (e.g.atomic.and
, etc.).atomic.store_add
: clearly states that this operation is a "store" (and not a "load").atomic.reduce_add
: clearly state what this operation is for (reductions).We considered providing separate version of non-associative floating-point atomic reduction operations with and without the support for tree-reductions, e.g.,
atomic.reduce_add
(no tree-reduction support) andatomic.reduce_add_generalized
(tree-reduction support), but decided against it because atomic operations are inherently non-deterministic, this relaxation only minimally impacts that, andfetch_add
already provides (and has to continue to provide) the sequential semantics without this relaxation.Memory Ordering
We choose to support
memory_order_relaxed
,memory_order_release
, andmemory_order_seq_cst
.We may need a note stating that replacing the operations in a reduction sequence is only valid as long as the replacement maintains the other ordering properties of the operations as defined in [intro.races]. Examples:
Litmus test 2:
Litmus test 3:
Unresolved question: Are tree reductions only supported for
memory_order_relaxed
? No, see litmus tests for release and seq_cst.Formalization
Herd already support these for
STADD
on Arm, and the NVIDIA Volta Memory Model supports these forred
andmultimem.red
on PTX. If we decide to pursue this exposure direction, this proposal would benefit from extending Herd's RC11 with reduction sequences for floating-point.Wording
Do NOT modify [intro.execution.10]!
We do not need to modify [intro.execution.10] to enable using atomic reduction operations in unsequenced contexts, because this section does not prevent that:
atomic.foo()
are function calls and two function calls are always indeterminately sequenced, not unsequenced. That is, function calls never overlap, and this section does not impact that.Add to [intro.races] before [intro.races.14]:
Review Note: the intent of this wording change is to enable tree-reduction implementations for non-associative floating-point atomic reduction operations which, in this proposal, are only
reduce_add
andreduce_sub
. SG1 agreed that the single floating point reduction operations should allow "fast math" (allow tree reductions) because programs can get precise results usingfetch_add
/sub
. Alternative wording strategies usingGENERALIZED_NONCOMMUTATIVE_SUM
were explored but are not viable since only two values are involved.Review Note: Since other proposals in flight add
fetch_mul
andfetch_div
, and those would need to extend this, this revision of this paper shows how that would look like by introducing "multiplicative reduction sequences".Review Note: For integers, implementations are already allowed to perform tree reductions, as long as they don't violate any other ordering rules. The only goal of the following reduction sequence wording is to allow the exact same thing for floats, since implementation cannot perform tree reductions because the ops are not associative and impact the results. Maybe instead of adding all of this wording, we could instead say that reduction operations on floats may be re-ordered or re-associated "as if" they were operations on integers? For example, we could add a Remark to the floating-point specialization of atomic and atomic_ref that just says: "Remark: The associativity of floating-point atomic reduction operations is the same as that of integers."
Option A.0: this option specifies the optimization that is allowed for floating-point atomic reduction operations..
[Note 1: Combining adjacent operations in a reduction sequence is often not observable for integer operations but is often observable for floating-point operations due to associativity. This paragraph enables implementations to optimize floating-point atomic reduction operations using tree reductions. - end note]
Alternative A.1: Replace the last sentence in the paragraph with: "Any two adjacent floating-point operations in a reduction sequence can be replaced by a single operation as if the operations were integer operations.".
Option B: this option explicitly specify the allowed replacements for floating-point operations using a table.
memory_order_relaxed < memory_order_release < memory_order_seq_cst
. For signed integer types, the intermediate computation is performed as if the objectsa
andb
were converted to their corresponding unsigned types, and its result converted back to the signed type.[Note 1: Replacing any two adjacent operations in a reduction sequence is only valid if the replacement maintains the other ordering properties of the operations as defined in [intro.races]. The assertion in the following example holds:
That is, replacing the two adjacent
M0.reduce_add
onthread0
with a singlereduce_add
after theM1
release store as follows:is not allowed, but replacing them before the store as follows:
Table [tab.red.seq.replacement]: Allowed replacements of two adjacent floating-point atomic reduction operations
reduce_<key>
within a floating-point atomic reduction sequence on memory location M, where the value parameter of the first atomic operation isa
, and that of the second isb
.First Op
add
sub
min
max
add
add(a + b)
add(a - b)
sub
add(b - a)
sub(a + b)
min
min(min(a,b))
max
max(max(a,b))
Option B.1: this option may be extended to mul and div operations as follows:
First Op
mul
div
mul
mul(a * b)
mul(a / b)
div
mul(b / a)
div(a * b)
Option B.2: this option explicitly specify the allowed replacements for floating-point operations without a table.
M.reduce_add(a, o0)
andM.reduce_add(b, o1)
can be replaced withM.reduce_add(a + b, max_order(o0, o1))
,M.reduce_sub(a, o0)
andM.reduce_sub(b, o1)
can be replaced withM.reduce_sub(a + b, max_order(o0, o1))
,M.reduce_add(a, o0)
andM.reduce_sub(b, o1)
can be replaced withM.reduce_add(a - b, max_order(o0, o1))
,M.reduce_sub(a, o0)
andM.reduce_add(b, o1)
can be replaced withM.reduce_add(b - a, max_order(o0, o1))
,M.reduce_mul(a, o0)
andM.reduce_mul(b, o1)
can be replaced withM.reduce_mul(a * b, max_order(o0, o1))
,M.reduce_div(a, o0)
andM.reduce_div(b, o1)
can be replaced withM.reduce_div(a * b, max_order(o0, o1))
,M.reduce_mul(a, o0)
andM.reduce_div(b, o1)
can be replaced withM.reduce_mul(a / b, max_order(o0, o1))
,M.reduce_div(a, o0)
andM.reduce_mul(b, o1)
can be replaced withM.reduce_mul(b / a, max_order(o0, o1))
,M.reduce_min(a, o0)
andM.reduce_min(b, o1)
can be replaced withM.reduce_min(min(a, b), max_order(o0, o1))
,M.reduce_max(a, o0)
andM.reduce_max(b, o1)
can be replaced withM.reduce_max(max(a, b), max_order(o0, o1))
.Add to [algorithms.parallel.defns]:
Review note: the first bullet says which standard library functions are, in general, vectorization unsafe, and the second bullet carves out exceptions.
Review note: the current wording intent is for
fetch_add(relaxed)
to be vectorization-unsafe, but the standard words in [intro.execution.10] (see above) that were supposed to fix that, do not. So we currently ban those here as a drive-by change that should be filled as LWG defect report, but that this proposal would then need to carve out an exception for atomic reduction operations.Review Note: SG1 requested making "relaxed", "release", and "seq_cst" atomic reduction operations not be vectorization unsafe, and that is the intent of the words below.
Forward progress
Modify [intro.progress#1] as follows:
Review Note: SG1 design intent is that Reduction operations are not a step, but infinite loops around reductions operations are not UB (practically implementations can assume reductions are finite). Gonzalo's note: such loops cause all threads to loose forward progress.
Review Note: this change makes Atomic Reduction Operations and Atomic Stores/Fences asymmetric, but making this change for Atomic Stores/Fences is a breaking change to be pursued in a separate paper per SG1 feedback. When adding these new operations, it does make sense to make this change, to avoid introducing undesired semantics that would be a breaking change to fix.
Modify [intro.progress#3] as follows:
Review note: same note as above about exempting atomic stores in a separate paper.
No acquire sequences support
Modify [atomics.fences] as follows:
Atomic Reduction Operation APIs
The following operations perform arithmetic computations. The correspondence among key, operator, and computation is specified in Table 145:
+
-
&
|
^
Add to [atomics.syn]:
Add to [atomics.ref.generic]:
Add to [atomics.ref.ops]:
order
ismemory_order::relaxed
,memory_order::release
, ormemory_order::seq_cst
.*ptr
for equality with the value inexpected
, and iftrue
, replaces the value pointed to by*ptr
with that indesired
. If and only if the comparison istrue
, memory is affected according to the value oforder
; otherwise, memory is not affected. This operation is an atomic read-modify-write operations ([intro.races]) and an atomic reduction operations on the value referenced by*ptr
.Add to [atomics.ref.int]:
order
ismemory_order_relaxed
,memory_order_release
, ormemory_order_seq_cst
.*ptr
with the result of the computation applied to the value referenced by*ptr
and the givenoperand
. Memory is affected according to the value oforder
. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.reduce_max
andreduce_min
, 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 2: There are no undefined results arising from the computation. — end note]
[Note 3: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note]
For
reduce_max
andreduce_min
, the maximum and minimum computation is performed as if bymax
andmin
algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.Add to [atomics.ref.float]:
Review note: add reduce_minimum, _minimum_num, _maximum, and _maximum_num once 3008 is merged.
order
ismemory_order_relaxed
,memory_order_release
, ormemory_order_seq_cst
.*ptr
with the result of the computation applied to the value referenced by*ptr
and the givenoperand
. Memory is affected according to the value oforder
. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.std::numeric_limits<floating-point-type>
traits associated with the floating-point type ([limits.syn]). The floating-point environment ([cfenv]) for atomic arithmetic operations on floating-point-type may be different than the calling thread's floating-point environment.Add to [atomics.ref.pointer]:
T
is a complete object type.ins>order
ismemory_order_relaxed
,memory_order_release
, ormemory_order_seq_cst
.*ptr
with the result of the computation applied to the value referenced by*ptr
and the given operand. Memory is affected according to the value of order. These operations are atomic read-modify-write operations ([intro.races]) and atomic reduction operations.For
fetch_max
andfetch_min
, the maximum and minimum computation is performed as if bymax
andmin
algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.[Note 1: If the pointers point to different complete objects (or subobjects thereof), the
<
operator does not establish a strict weak ordering (Table 29, [expr.rel]). — end note][Note 2: A lock-free atomic reduction operation is not vectorization-unsafe ([algorithms.parallel.defns]). — end note]
Add to [atomics.types.generic.general]:
Add to [atomics.types.operations]:
order
ismemory_order::relaxed
,memory_order::release
, ormemory_order::seq_cst
.this
for equality with that ofexpected
, and iftrue
, replaces the value pointed to bythis
with that indesired
. If and only if the comparison istrue
, memory is affected according to the value oforder
; otherwise, memory is not affected. This operation is an atomic read-modify-write operations ([intro.races]) and an atomic reduction operations on the memory pointed to bythis
.Add to [atomics.types.int]:
volatile
overload of this function,is_always_lock_free
is true.order
ismemory_order_relaxed
,memory_order_release
, ormemory_order_seq_cst
.this
with the result of the computation applied to the value pointed to bythis
and the given operand. Memory is affected according to the value oforder
. These operations are atomic read-modify-write operations ([intro.multithread]) and atomic reduction operations.reduce_max
andreduce_min
, 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 2: There are no undefined results arising from the computation. — end note]
For
reduce_max
andreduce_min
, the maximum and minimum computation is performed as if bymax
andmin
algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.Add to [atomics.types.float]:
Review note: add reduce_minimum, _minimum_num, _maximum, and _maximum_num once 3008 is merged.
volatile
overload of this function,is_always_lock_free
is true.order
ismemory_order_relaxed
,memory_order_release
, ormemory_order_seq_cst
.this
with the result of the computation applied to the value pointed to bythis
and the givenoperand
. Memory is affected according to the value oforder
. These operations are atomic read-modify-write operations ([intro.multithread]) and atomic reduction operations.floating-point-type
should conform to thestd::numeric_limits<floating-point-type>
traits associated with the floating-point type ([limits.syn]). The floating-point environment ([cfenv]) for atomic arithmetic operations onfloating-point-type
may be different than the calling thread's floating-point environment.Add to [atomics.types.pointer]:
volatile
overload of this function,is_always_lock_free
is true.T
is a complete object type.[Note 1: Pointer arithmetic on void* or function pointers is ill-formed. — end note]
this
and the given operand. Memory is affected according to the value oforder
. These operations are atomic read-modify-write operations ([intro.multithread]).For
reduce_max
andreduce_min
, the maximum and minimum computation is performed as if bymax
andmin
algorithms ([alg.min.max]), respectively, with the object value and the first parameter as the arguments.[Note 2: If the pointers point to different complete objects (or subobjects thereof), the
<
operator does not establish a strict weak ordering (Table 29, [expr.rel]). — end note]Add to [util.smartptr.atomic.shared]:
Add to [util.smartptr.atomic.weak]: