Fix CompositeMutex concurrent locking
This commit is contained in:
@@ -1,6 +1,8 @@
|
|||||||
package org.koitharu.kotatsu.utils
|
package org.koitharu.kotatsu.utils
|
||||||
|
|
||||||
import kotlinx.coroutines.CancellableContinuation
|
import kotlinx.coroutines.CancellableContinuation
|
||||||
|
import kotlinx.coroutines.currentCoroutineContext
|
||||||
|
import kotlinx.coroutines.isActive
|
||||||
import kotlinx.coroutines.suspendCancellableCoroutine
|
import kotlinx.coroutines.suspendCancellableCoroutine
|
||||||
import kotlinx.coroutines.sync.Mutex
|
import kotlinx.coroutines.sync.Mutex
|
||||||
import kotlinx.coroutines.sync.withLock
|
import kotlinx.coroutines.sync.withLock
|
||||||
@@ -32,11 +34,13 @@ class CompositeMutex<T : Any> : Set<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
suspend fun lock(element: T) {
|
suspend fun lock(element: T) {
|
||||||
waitForRemoval(element)
|
while (currentCoroutineContext().isActive) {
|
||||||
mutex.withLock {
|
waitForRemoval(element)
|
||||||
val lastValue = data.put(element, LinkedList<CancellableContinuation<Unit>>())
|
mutex.withLock {
|
||||||
check(lastValue == null) {
|
if (data[element] == null) {
|
||||||
"CompositeMutex is double-locked for $element"
|
data[element] = LinkedList<CancellableContinuation<Unit>>()
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
package org.koitharu.kotatsu.utils
|
||||||
|
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
import kotlinx.coroutines.test.runTest
|
||||||
|
import kotlinx.coroutines.withTimeoutOrNull
|
||||||
|
import kotlinx.coroutines.yield
|
||||||
|
import org.junit.Assert.assertNull
|
||||||
|
import org.junit.Test
|
||||||
|
|
||||||
|
class CompositeMutexTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testSingleLock() = runTest {
|
||||||
|
val mutex = CompositeMutex<Int>()
|
||||||
|
mutex.lock(1)
|
||||||
|
mutex.lock(2)
|
||||||
|
mutex.unlock(1)
|
||||||
|
assert(mutex.size == 1)
|
||||||
|
mutex.unlock(2)
|
||||||
|
assert(mutex.isEmpty())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testDoubleLock() = runTest {
|
||||||
|
val mutex = CompositeMutex<Int>()
|
||||||
|
repeat(2) {
|
||||||
|
launch(Dispatchers.Default) {
|
||||||
|
mutex.lock(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
yield()
|
||||||
|
mutex.unlock(1)
|
||||||
|
val tryLock = withTimeoutOrNull(1000) {
|
||||||
|
mutex.lock(1)
|
||||||
|
}
|
||||||
|
assertNull(tryLock)
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user