All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.flow.internal.Concurrent.kt Maven / Gradle / Ivy

/*
 * Copyright 2016-2019 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
 */

package kotlinx.coroutines.flow.internal

import kotlinx.atomicfu.*
import kotlinx.coroutines.*
import kotlinx.coroutines.channels.*
import kotlinx.coroutines.channels.ArrayChannel
import kotlinx.coroutines.flow.*

internal fun  FlowCollector.asConcurrentFlowCollector(): ConcurrentFlowCollector =
    this as? ConcurrentFlowCollector ?: SerializingCollector(this)

// Flow collector that supports concurrent emit calls.
// It is internal for now but may be public in the future.
// Two basic implementations are here: SendingCollector and ConcurrentFlowCollector
internal interface ConcurrentFlowCollector : FlowCollector

/**
 * Collection that sends to channel. It is marked as [ConcurrentFlowCollector] because it can be used concurrently.
 *
 * @suppress **This an internal API and should not be used from general code.**
 */
@InternalCoroutinesApi
public class SendingCollector(
    private val channel: SendChannel
) : ConcurrentFlowCollector {
    override suspend fun emit(value: T) = channel.send(value)
}

// Effectively serializes access to downstream collector for merging
// This is basically a converted from FlowCollector interface to ConcurrentFlowCollector
private class SerializingCollector(
    private val downstream: FlowCollector
) : ConcurrentFlowCollector {
    // Let's try to leverage the fact that merge is never contended
    // Should be Any, but KT-30796
    private val _channel = atomic?>(null)
    private val inProgressLock = atomic(false)

    private val channel: ArrayChannel
        get() = _channel.updateAndGet { value ->
            if (value != null) return value
            ArrayChannel(Channel.CHANNEL_DEFAULT_CAPACITY)
        }!!

    public override suspend fun emit(value: T) {
        if (!inProgressLock.tryAcquire()) {
            channel.send(value ?: NULL)
            if (inProgressLock.tryAcquire()) {
                helpEmit()
            }
            return
        }
        downstream.emit(value)
        helpEmit()
    }

    @Suppress("UNCHECKED_CAST")
    private suspend fun helpEmit() {
        while (true) {
            while (true) {
                val element = _channel.value?.poll() ?: break // todo: pollOrClosed
                downstream.emit(NULL.unbox(element))
            }
            inProgressLock.release()
            // Enforce liveness
            if (_channel.value?.isEmpty != false || !inProgressLock.tryAcquire()) break
        }
    }
}

@Suppress("NOTHING_TO_INLINE")
private inline fun AtomicBoolean.tryAcquire(): Boolean = compareAndSet(false, true)

@Suppress("NOTHING_TO_INLINE")
private inline fun AtomicBoolean.release() {
    value = false
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy