The atomic module provides basic support for lock-free concurrent programming.
Specifies the memory ordering semantics of an atomic operation.
pure nothrow @nogc @trusted T
atomicLoad(MemoryOrder ms = MemoryOrder.seq, T)(auto ref return scope const T
val)
if (!is(T == shared(U), U) && !is(T == shared(inout(U)), U) && !is(T == shared(const(U)), U));
pure nothrow @nogc @trusted T
atomicLoad(MemoryOrder ms = MemoryOrder.seq, T)(auto ref return scope const shared T
val)
if (!hasUnsharedIndirections!T);
pure nothrow @nogc @trusted TailShared!T
atomicLoad(MemoryOrder ms = MemoryOrder.seq, T)(auto ref const shared T
val)
if (hasUnsharedIndirections!T);
Loads 'val' from memory and returns it. The memory barrier specified by 'ms' is applied to the operation, which is fully sequenced by default. Valid memory orders are MemoryOrder.raw, MemoryOrder.acq, and MemoryOrder.seq.
Returns:The value of 'val'.
pure nothrow @nogc @trusted void
atomicStore(MemoryOrder ms = MemoryOrder.seq, T, V)(ref T
val, V
newval)
if (!is(T == shared) && !is(V == shared));
pure nothrow @nogc @trusted void
atomicStore(MemoryOrder ms = MemoryOrder.seq, T, V)(ref shared T
val, V
newval)
if (!is(T == class));
pure nothrow @nogc @trusted void
atomicStore(MemoryOrder ms = MemoryOrder.seq, T, V)(ref shared T
val, auto ref shared V
newval)
if (is(T == class));
Writes 'newval' into 'val'. The memory barrier specified by 'ms' is applied to the operation, which is fully sequenced by default. Valid memory orders are MemoryOrder.raw, MemoryOrder.rel, and MemoryOrder.seq.
Parameters:Tval | The target variable. |
Vnewval | The value to store. |
pure nothrow @nogc @trusted T
atomicFetchAdd(MemoryOrder ms = MemoryOrder.seq, T)(ref return scope T
val, size_t
mod)
if ((__traits(isIntegral, T) || is(T == U*, U)) && !is(T == shared));
pure nothrow @nogc @trusted T
atomicFetchAdd(MemoryOrder ms = MemoryOrder.seq, T)(ref return scope shared T
val, size_t
mod)
if (__traits(isIntegral, T) || is(T == U*, U));
Atomically addsmod to the value referenced byval and returns the valueval held previously. This operation is both lock-free and atomic.
Parameters:Tval | Reference to the value to modify. |
size_tmod | The value to add. |
Returns:The value held previously byval.
pure nothrow @nogc @trusted T
atomicFetchSub(MemoryOrder ms = MemoryOrder.seq, T)(ref return scope T
val, size_t
mod)
if ((__traits(isIntegral, T) || is(T == U*, U)) && !is(T == shared));
pure nothrow @nogc @trusted T
atomicFetchSub(MemoryOrder ms = MemoryOrder.seq, T)(ref return scope shared T
val, size_t
mod)
if (__traits(isIntegral, T) || is(T == U*, U));
Atomically subtractsmod from the value referenced byval and returns the valueval held previously. This operation is both lock-free and atomic.
Parameters:Tval | Reference to the value to modify. |
size_tmod | The value to subtract. |
Returns:The value held previously byval.
pure nothrow @nogc @trusted T
atomicExchange(MemoryOrder ms = MemoryOrder.seq, T, V)(T*
here, V
exchangeWith)
if (!is(T == shared) && !is(V == shared));
pure nothrow @nogc @trusted TailShared!T
atomicExchange(MemoryOrder ms = MemoryOrder.seq, T, V)(shared(T)*
here, V
exchangeWith)
if (!is(T == class) && !is(T == interface));
pure nothrow @nogc @trusted shared(T)
atomicExchange(MemoryOrder ms = MemoryOrder.seq, T, V)(shared(T)*
here, shared(V)
exchangeWith)
if (is(T == class) || is(T == interface));
ExchangeexchangeWith with the memory referenced byhere. This operation is both lock-free and atomic.
Parameters:T*here | The address of the destination variable. |
VexchangeWith | The value to exchange. |
Returns:The value held previously byhere.
template
cas(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq)
Performs either compare-and-set or compare-and-swap (or exchange).
There are two categories of overloads in this template: The first category does a simple compare-and-set. The comparison value (
ifThis) is treated as an rvalue.
The second category does a compare-and-swap (a.k.a. compare-and-exchange), and expects
ifThis to be a pointer type, where the previous value of
here will be written.
This operation is both lock-free and atomic.
Parameters:| T* here | The address of the destination variable. |
| V2 writeThis | The value to store. |
| V1 ifThis | The comparison value. |
Returns:true if the store occurred, false if not.
pure nothrow @nogc @trusted bool
cas(T, V1, V2)(T*
here, V1
ifThis, V2
writeThis)
if (!is(T == shared) && is(T : V1));
Compare-and-set for non-shared values
pure nothrow @nogc @trusted bool
cas(T, V1, V2)(shared(T)*
here, V1
ifThis, V2
writeThis)
if (!is(T == class) && (is(T : V1) || is(shared(T) : V1)));
Compare-and-set for shared value type
pure nothrow @nogc @trusted bool
cas(T, V1, V2)(shared(T)*
here, shared(V1)
ifThis, shared(V2)
writeThis)
if (is(T == class));
Compare-and-set forshared reference type (class)
pure nothrow @nogc @trusted bool
cas(T, V)(T*
here, T*
ifThis, V
writeThis)
if (!is(T == shared) && !is(V == shared));
Compare-and-exchange for non-shared types
pure nothrow @nogc @trusted bool
cas(T, V1, V2)(shared(T)*
here, V1*
ifThis, V2
writeThis)
if (!is(T == class) && (is(T : V1) || is(shared(T) : V1)));
Compare and exchange for mixed-sharedness types
pure nothrow @nogc @trusted bool
cas(T, V)(shared(T)*
here, shared(T)*
ifThis, shared(V)
writeThis)
if (is(T == class));
Compare-and-exchange forclass
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V1, V2)(T*
here, V1
ifThis, V2
writeThis)
if (!is(T == shared) && is(T : V1));
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V1, V2)(shared(T)*
here, V1
ifThis, V2
writeThis)
if (!is(T == class) && (is(T : V1) || is(shared(T) : V1)));
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V1, V2)(shared(T)*
here, shared(V1)
ifThis, shared(V2)
writeThis)
if (is(T == class));
Stores 'writeThis' to the memory referenced by 'here' if the value referenced by 'here' is equal to 'ifThis'. The 'weak' version of cas may spuriously fail. It is recommended to usecasWeak only whencas would be used in a loop. This operation is both lock-free and atomic.
Parameters:T*here | The address of the destination variable. |
V2writeThis | The value to store. |
V1ifThis | The comparison value. |
Returns:true if the store occurred, false if not.
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V)(T*
here, T*
ifThis, V
writeThis)
if (!is(T == shared(S), S) && !is(V == shared(U), U));
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V1, V2)(shared(T)*
here, V1*
ifThis, V2
writeThis)
if (!is(T == class) && (is(T : V1) || is(shared(T) : V1)));
pure nothrow @nogc @trusted bool
casWeak(MemoryOrder succ = MemoryOrder.seq, MemoryOrder fail = MemoryOrder.seq, T, V)(shared(T)*
here, shared(T)*
ifThis, shared(V)
writeThis)
if (is(T == class));
Stores 'writeThis' to the memory referenced by 'here' if the value referenced by 'here' is equal to the value referenced by 'ifThis'. The prior value referenced by 'here' is written toifThis and returned to the user. The 'weak' version of cas may spuriously fail. It is recommended to usecasWeak only whencas would be used in a loop. This operation is both lock-free and atomic.
Parameters:T*here | The address of the destination variable. |
VwriteThis | The value to store. |
T*ifThis | The address of the value to compare, and receives the prior value ofhere as output. |
Returns:true if the store occurred, false if not.
pure nothrow @nogc @safe void
atomicFence(MemoryOrder order = MemoryOrder.seq)();
Inserts a full load/store memory fence (on platforms that need it). This ensures that all loads and stores before a call to this function are executed before any loads and stores after the call.
pure nothrow @nogc @safe void
pause();
Gives a hint to the processor that the calling thread is in a 'spin-wait' loop, allowing to more efficiently allocate resources.
pure nothrow @nogc @safe TailShared!T
atomicOp(string op, T, V1)(ref shared T
val, V1
mod)
if (__traits(compiles, mixin("*cast(T*)&val" ~ op ~ "mod")));
Performs the binary operation 'op' on val using 'mod' as the modifier.
Parameters:Tval | The target variable. |
V1mod | The modifier to apply. |
Returns:The result of the operation.