-
Notifications
You must be signed in to change notification settings - Fork 1.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
New select
and Mutex
algorithms
#3020
Conversation
select
implementationselect
implementation
23e9ff2
to
b43b7bb
Compare
927ebb6
to
b9b8dc4
Compare
f3c5772
to
b2beac6
Compare
select
implementationselect
and Mutex
algorithms
75a9947
to
e38ca48
Compare
kotlinx-coroutines-core/common/src/internal/LocalAtomics.common.kt
Outdated
Show resolved
Hide resolved
kotlinx-coroutines-core/common/src/internal/LockFreeLinkedList.common.kt
Outdated
Show resolved
Hide resolved
Also, ChannelUndeliveredElementSelectOldStressTest fails, please fix that as well, we would like to avoid having any flaking tests in the mainline |
As for the flacking test, there was a problem in the test itself. Now the tests are green. |
42f24b7
to
df3aae1
Compare
TODO for me: cleanup documentation when it's the time |
@@ -592,7 +593,6 @@ public abstract interface class kotlinx/coroutines/channels/ActorScope : kotlinx | |||
|
|||
public final class kotlinx/coroutines/channels/ActorScope$DefaultImpls { | |||
public static synthetic fun cancel (Lkotlinx/coroutines/channels/ActorScope;)V | |||
public static fun getOnReceiveOrNull (Lkotlinx/coroutines/channels/ActorScope;)Lkotlinx/coroutines/selects/SelectClause1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same here, should be left as is
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's non-trivial to have a default implementation. I would suggest performing a cast to AbstractChannel and invoking onReceiveOrNull
after that, throwing an exception if the channel implementation is not ours.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I will do an unchecked cast to AbstractChannel
, we have discussed this in Slack
kotlinx-coroutines-core/jvm/test/exceptions/StackTraceRecoverySelectTest.kt
Outdated
Show resolved
Hide resolved
* | ||
* @suppress **This is unstable API and it is subject to change.** | ||
* @suppress **This is unstable API, and it is subject to change.** | ||
*/ | ||
@InternalCoroutinesApi // todo: sealed interface https://youtrack.jetbrains.com/issue/KT-22286 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point, will leave it as a documentation improvement
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also, please revisit SelectInstance
being sealed as Oleg suggested
if (!_state.compareAndSet(NOT_SELECTED, pairSelectOp)) return@loop | ||
val decision = pairSelectOp.perform(this) | ||
if (decision !== null) return decision | ||
fun trySelectDetailed(clauseObject: Any, result: Any?) = |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The only use of this function is tryResumeSend
which only cares about whether it's success or not (=> should rely on trySelect
).
If you are not using this function in new channels, please remove it and all around it
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The current implementation performs the following check, so it cannot be replaced with trySelect
:
send.trySelectResult == REREGISTER
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks
if (segment.cas(i, null, cont)) { // installed continuation successfully | ||
cont.invokeOnCancellation(CancelSemaphoreAcquisitionHandler(segment, i).asHandler) | ||
if (segment.cas(i, null, waiter)) { // installed continuation successfully | ||
when (waiter) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think that the code would be much cleaner if we introduced a private sealed interface to represent the possible types of waiters. This would cost us one additional application per slow-path acquire, but get rid of Any
, impossible error
, etc.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree with you, but your suggestion requires either CancellableContinuation
to implement this sealed interface or adding unsafe casts at every CancellableContinuation
usage (in case onlyCancellableContinuationImpl
implements the interface). I do not see a good solution, so I would keep the current code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Dima, thanks, it's indeed worth doing. I'll try to do it separately in a branch with new channels, so the disturbance will be minimal. I have a prototype for selects only and it looks much more readable
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The idea was to do something like this (pseudocode):
private sealed interface ResumableComputation {
fun resume()
fun disposeOnCancellation(handle: DisposableHandle)
}
private class CancellableContinuationAsResumable(
val continuation: CancellableContinuation<Unit>
): ResumableComputation {
...
}
So, CancellableContinuation
would not itself implement anything (though maybe this interface makes sense in general and it should), it would be wrapped in an adapter. The cost is one more allocation on the slow path.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It is better not to do this. With #3084 and continuation reusability, the suspension becomes exceptionally efficient, as it does not allocate new objects. The change you suggest breaks this property, evolving extra work for GC.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please address Dmitry's comments and it's good to go
c7eb4dc
to
f0c3f5a
Compare
see the documentation in `Select.kt` and `Mutex.kt` files for details
f0c3f5a
to
851cb9b
Compare
I went through Dmitry's comments and checked the ABI manually. |
👏 |
…block throws an exception Otherwise, continuation instance is left in REUSABLE_CLAIMED state that asynchronous resumer awaits in an infinite spin-loop, potentially causing deadlock with 100% CPU consumption. Originally, the bug was reproduced on old (pre-#3020) implementation where this very pattern was encountered: it was possible to fail owner's invariant check right in the supplied 'block'. This is no longer the case, so the situation is emulated manually (but still is possible in production environments, e.g. when OOM is thrown). Also, suspendCancellableCoroutineReusable is removed from obsolete BroadcastChannel implementation. Fixes #3613
#3634) * Release reusability token when suspendCancellableCoroutineReusable's block throws an exception Otherwise, the continuation instance is left in REUSABLE_CLAIMED state that asynchronous resumer awaits in an infinite spin-loop, potentially causing deadlock with 100% CPU consumption. Originally, the bug was reproduced on an old (pre-#3020) implementation where this very pattern was encountered: it was possible to fail the owner's invariant check right in the supplied 'block'. This is no longer the case, so the situation is emulated manually (but still is possible in production environments, e.g. when OOM is thrown). Also, suspendCancellableCoroutineReusable is removed from obsolete BroadcastChannel implementation. Fixes #3613
New
select
andMutex
algorithms