commonMain.kotlinx.atomicfu.AtomicFU.common.kt Maven / Gradle / Ivy
/*
* Copyright 2017-2018 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
*/
@file:Suppress("INVISIBLE_REFERENCE", "INVISIBLE_MEMBER")
package kotlinx.atomicfu
import kotlin.internal.InlineOnly
import kotlinx.atomicfu.TraceBase.None
import kotlin.reflect.KProperty
/**
* Creates atomic reference with a given [initial] value and a [trace] object to [trace modifications][Trace] of the value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initial, trace)
* ```
*/
public expect fun atomic(initial: T, trace: TraceBase = None): AtomicRef
// Binary compatibility with IR, should be removed with Kotlin 1.5 release
/**
* Creates atomic reference with a given [initial] value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initial)
* ```
*/
public expect fun atomic(initial: T): AtomicRef
/**
* Creates atomic [Int] with a given [initial] value and a [trace] object to [trace modifications][Trace] of the value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialInt, trace)
* ```
*/
public expect fun atomic(initial: Int, trace: TraceBase = None): AtomicInt
// Binary compatibility with IR, should be removed with Kotlin 1.5 release
/**
* Creates atomic [Int] with a given [initial] value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialInt)
* ```
*/
public expect fun atomic(initial: Int): AtomicInt
/**
* Creates atomic [Long] with a given [initial] value and a [trace] object to [trace modifications][Trace] of the value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialLong, trace)
* ```
*/
public expect fun atomic(initial: Long, trace: TraceBase = None): AtomicLong
// Binary compatibility with IR, should be removed with Kotlin 1.5 release
/**
* Creates atomic [Long] with a given [initial] value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialLong)
* ```
*/
public expect fun atomic(initial: Long): AtomicLong
/**
* Creates atomic [Boolean] with a given [initial] value and a [trace] object to [trace modifications][Trace] of the value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialBoolean, trace)
* ```
*/
public expect fun atomic(initial: Boolean, trace: TraceBase = None): AtomicBoolean
// Binary compatibility with IR, should be removed with Kotlin 1.5 release
/**
* Creates atomic [Boolean] with a given [initial] value.
*
* It can only be used to initialize a private or internal read-only property, like this:
*
* ```
* private val f = atomic(initialBoolean)
* ```
*/
public expect fun atomic(initial: Boolean): AtomicBoolean
/**
* Creates array of AtomicRef of specified size, where each element is initialised with null value
*/
@OptionalJsName(ATOMIC_ARRAY_OF_NULLS)
public fun atomicArrayOfNulls(size: Int): AtomicArray = AtomicArray(size)
// ==================================== AtomicRef ====================================
/**
* Atomic reference to a variable of type [T] with volatile reads/writes via
* [value] property and various atomic read-modify-write operations
* like [compareAndSet] and others.
*/
public expect class AtomicRef {
/**
* Reading/writing this property maps to read/write of volatile variable.
*/
public var value: T
@InlineOnly
public inline operator fun getValue(thisRef: Any?, property: KProperty<*>): T
@InlineOnly
public inline operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T)
/**
* Maps to [AtomicReferenceFieldUpdater.lazySet].
*/
public fun lazySet(value: T)
/**
* Maps to [AtomicReferenceFieldUpdater.compareAndSet].
*/
public fun compareAndSet(expect: T, update: T): Boolean
/**
* Maps to [AtomicReferenceFieldUpdater.getAndSet].
*/
public fun getAndSet(value: T): T
}
/**
* Infinite loop that reads this atomic variable and performs the specified [action] on its value.
*/
public inline fun AtomicRef.loop(action: (T) -> Unit): Nothing {
while (true) {
action(value)
}
}
/**
* Updates variable atomically using the specified [function] of its value.
*/
public inline fun AtomicRef.update(function: (T) -> T) {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its old value.
*/
public inline fun AtomicRef.getAndUpdate(function: (T) -> T): T {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return cur
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its new value.
*/
public inline fun AtomicRef.updateAndGet(function: (T) -> T): T {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return upd
}
}
// ==================================== AtomicBoolean ====================================
/**
* Atomic reference to a [Boolean] variable with volatile reads/writes via
* [value] property and various atomic read-modify-write operations
* like [compareAndSet] and others.
*/
public expect class AtomicBoolean {
/**
* Reading/writing this property maps to read/write of volatile variable.
*/
public var value: Boolean
@InlineOnly
public inline operator fun getValue(thisRef: Any?, property: KProperty<*>): Boolean
@InlineOnly
public inline operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Boolean)
/**
* Maps to [AtomicIntegerFieldUpdater.lazySet].
*/
public fun lazySet(value: Boolean)
/**
* Maps to [AtomicIntegerFieldUpdater.compareAndSet].
*/
public fun compareAndSet(expect: Boolean, update: Boolean): Boolean
/**
* Maps to [AtomicIntegerFieldUpdater.getAndSet].
*/
public fun getAndSet(value: Boolean): Boolean
}
/**
* Infinite loop that reads this atomic variable and performs the specified [action] on its value.
*/
public inline fun AtomicBoolean.loop(action: (Boolean) -> Unit): Nothing {
while (true) {
action(value)
}
}
/**
* Updates variable atomically using the specified [function] of its value.
*/
public inline fun AtomicBoolean.update(function: (Boolean) -> Boolean) {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its old value.
*/
public inline fun AtomicBoolean.getAndUpdate(function: (Boolean) -> Boolean): Boolean {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return cur
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its new value.
*/
public inline fun AtomicBoolean.updateAndGet(function: (Boolean) -> Boolean): Boolean {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return upd
}
}
// ==================================== AtomicInt ====================================
/**
* Atomic reference to an [Int] variable with volatile reads/writes via
* [value] property and various atomic read-modify-write operations
* like [compareAndSet] and others.
*/
public expect class AtomicInt {
/**
* Reads/writes of this property maps to read/write of volatile variable.
*/
public var value: Int
@InlineOnly
public inline operator fun getValue(thisRef: Any?, property: KProperty<*>): Int
@InlineOnly
public inline operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Int)
/**
* Maps to [AtomicIntegerFieldUpdater.lazySet].
*/
public fun lazySet(value: Int)
/**
* Maps to [AtomicIntegerFieldUpdater.compareAndSet].
*/
public fun compareAndSet(expect: Int, update: Int): Boolean
/**
* Maps to [AtomicIntegerFieldUpdater.getAndSet].
*/
public fun getAndSet(value: Int): Int
/**
* Maps to [AtomicIntegerFieldUpdater.getAndIncrement].
*/
public fun getAndIncrement(): Int
/**
* Maps to [AtomicIntegerFieldUpdater.getAndDecrement].
*/
public fun getAndDecrement(): Int
/**
* Maps to [AtomicIntegerFieldUpdater.getAndAdd].
*/
public fun getAndAdd(delta: Int): Int
/**
* Maps to [AtomicIntegerFieldUpdater.addAndGet].
*/
public fun addAndGet(delta: Int): Int
/**
* Maps to [AtomicIntegerFieldUpdater.incrementAndGet].
*/
public fun incrementAndGet(): Int
/**
* Maps to [AtomicIntegerFieldUpdater.decrementAndGet].
*/
public fun decrementAndGet(): Int
/**
* Performs atomic addition of [delta].
*/
public inline operator fun plusAssign(delta: Int)
/**
* Performs atomic subtraction of [delta].
*/
public inline operator fun minusAssign(delta: Int)
}
/**
* Infinite loop that reads this atomic variable and performs the specified [action] on its value.
*/
public inline fun AtomicInt.loop(action: (Int) -> Unit): Nothing {
while (true) {
action(value)
}
}
/**
* Updates variable atomically using the specified [function] of its value.
*/
public inline fun AtomicInt.update(function: (Int) -> Int) {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its old value.
*/
public inline fun AtomicInt.getAndUpdate(function: (Int) -> Int): Int {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return cur
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its new value.
*/
public inline fun AtomicInt.updateAndGet(function: (Int) -> Int): Int {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return upd
}
}
// ==================================== AtomicLong ====================================
/**
* Atomic reference to a [Long] variable with volatile reads/writes via
* [value] property and various atomic read-modify-write operations
* like [compareAndSet] and others.
*/
public expect class AtomicLong {
/**
* Reads/writes of this property maps to read/write of volatile variable.
*/
public var value: Long
@InlineOnly
public operator fun getValue(thisRef: Any?, property: KProperty<*>): Long
@InlineOnly
public operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Long)
/**
* Maps to [AtomicLongFieldUpdater.lazySet].
*/
public fun lazySet(value: Long)
/**
* Maps to [AtomicLongFieldUpdater.compareAndSet].
*/
public fun compareAndSet(expect: Long, update: Long): Boolean
/**
* Maps to [AtomicLongFieldUpdater.getAndSet].
*/
public fun getAndSet(value: Long): Long
/**
* Maps to [AtomicLongFieldUpdater.getAndIncrement].
*/
public fun getAndIncrement(): Long
/**
* Maps to [AtomicLongFieldUpdater.getAndDecrement].
*/
public fun getAndDecrement(): Long
/**
* Maps to [AtomicLongFieldUpdater.getAndAdd].
*/
public fun getAndAdd(delta: Long): Long
/**
* Maps to [AtomicLongFieldUpdater.addAndGet].
*/
public fun addAndGet(delta: Long): Long
/**
* Maps to [AtomicLongFieldUpdater.incrementAndGet].
*/
public fun incrementAndGet(): Long
/**
* Maps to [AtomicLongFieldUpdater.decrementAndGet].
*/
public fun decrementAndGet(): Long
/**
* Performs atomic addition of [delta].
*/
public inline operator fun plusAssign(delta: Long)
/**
* Performs atomic subtraction of [delta].
*/
public inline operator fun minusAssign(delta: Long)
}
/**
* Infinite loop that reads this atomic variable and performs the specified [action] on its value.
*/
public inline fun AtomicLong.loop(action: (Long) -> Unit): Nothing {
while (true) {
action(value)
}
}
/**
* Updates variable atomically using the specified [function] of its value.
*/
public inline fun AtomicLong.update(function: (Long) -> Long) {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its old value.
*/
public inline fun AtomicLong.getAndUpdate(function: (Long) -> Long): Long {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return cur
}
}
/**
* Updates variable atomically using the specified [function] of its value and returns its new value.
*/
public inline fun AtomicLong.updateAndGet(function: (Long) -> Long): Long {
while (true) {
val cur = value
val upd = function(cur)
if (compareAndSet(cur, upd)) return upd
}
}
// ==================================== AtomicIntArray ====================================
/**
* Creates a new array of AtomicInt values of the specified size, where each element is initialised with 0
*/
@OptionalJsName(ATOMIC_INT_ARRAY)
public class AtomicIntArray(size: Int) {
private val array = Array(size) { atomic(0) }
@OptionalJsName(ARRAY_SIZE)
public val size: Int
get() = array.size
@OptionalJsName(ARRAY_ELEMENT_GET)
public operator fun get(index: Int): AtomicInt = array[index]
}
// ==================================== AtomicLongArray ====================================
/**
* Creates a new array of AtomicLong values of the specified size, where each element is initialised with 0L
*/
@OptionalJsName(ATOMIC_LONG_ARRAY)
public class AtomicLongArray(size: Int) {
private val array = Array(size) { atomic(0L) }
@OptionalJsName(ARRAY_SIZE)
public val size: Int
get() = array.size
@OptionalJsName(ARRAY_ELEMENT_GET)
public operator fun get(index: Int): AtomicLong = array[index]
}
// ==================================== AtomicBooleanArray ====================================
/**
* Creates a new array of AtomicBoolean values of the specified size, where each element is initialised with false
*/
@OptionalJsName(ATOMIC_BOOLEAN_ARRAY)
public class AtomicBooleanArray(size: Int) {
private val array = Array(size) { atomic(false) }
@OptionalJsName(ARRAY_SIZE)
public val size: Int
get() = array.size
@OptionalJsName(ARRAY_ELEMENT_GET)
public operator fun get(index: Int): AtomicBoolean = array[index]
}
// ==================================== AtomicArray ====================================
@OptionalJsName(ATOMIC_REF_ARRAY)
public class AtomicArray internal constructor(size: Int) {
private val array = Array(size) { atomic(null) }
@OptionalJsName(ARRAY_SIZE)
public val size: Int
get() = array.size
@OptionalJsName(ARRAY_ELEMENT_GET)
public operator fun get(index: Int): AtomicRef = array[index]
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy