59 lines
1.5 KiB
Kotlin
Vendored
59 lines
1.5 KiB
Kotlin
Vendored
// TODO(KT-65977): reenable these tests with caches
|
|
//IGNORE_NATIVE: cacheMode=STATIC_EVERYWHERE
|
|
//IGNORE_NATIVE: cacheMode=STATIC_PER_FILE_EVERYWHERE
|
|
import kotlinx.atomicfu.*
|
|
import kotlin.test.*
|
|
|
|
class LockFreeQueueTest {
|
|
fun testBasic() {
|
|
val q = LockFreeQueue()
|
|
assertEquals(-1, q.dequeue())
|
|
q.enqueue(42)
|
|
assertEquals(42, q.dequeue())
|
|
assertEquals(-1, q.dequeue())
|
|
q.enqueue(1)
|
|
q.enqueue(2)
|
|
assertEquals(1, q.dequeue())
|
|
assertEquals(2, q.dequeue())
|
|
assertEquals(-1, q.dequeue())
|
|
}
|
|
}
|
|
|
|
// MS-queue
|
|
public class LockFreeQueue {
|
|
private val head = atomic(Node(0))
|
|
private val tail = atomic(head.value)
|
|
|
|
private class Node(val value: Int) {
|
|
val next = atomic<Node?>(null)
|
|
}
|
|
|
|
public fun enqueue(value: Int) {
|
|
val node = Node(value)
|
|
tail.loop { curTail ->
|
|
val curNext = curTail.next.value
|
|
if (curNext != null) {
|
|
tail.compareAndSet(curTail, curNext)
|
|
return@loop
|
|
}
|
|
if (curTail.next.compareAndSet(null, node)) {
|
|
tail.compareAndSet(curTail, node)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
public fun dequeue(): Int {
|
|
head.loop { curHead ->
|
|
val next = curHead.next.value ?: return -1
|
|
if (head.compareAndSet(curHead, next)) return next.value
|
|
}
|
|
}
|
|
}
|
|
|
|
@Test
|
|
fun box() {
|
|
val testClass = LockFreeQueueTest()
|
|
testClass.testBasic()
|
|
}
|